0%

mybatis之DataSource

DataSource数据源

所有的数据源组件都实现了javax.sql.DataSource接口,Mybatis实现了两个接口实现,分别为PooledDataSource和UnpooledDataSource

UnpooledDataSource

每次通过UnpooledDataSource.getConnection()方法获取数据库连接时都会创建一个新连接。

在UnpooledDataSource加载时首先会通过以下代码将已经在DriverManger中注册的JDBC驱动复制一份存到registeredDrivers中

1
2
3
4
5
6
7
8
9
static {
Enumeration drivers = DriverManager.getDrivers();

while(drivers.hasMoreElements()) {
Driver driver = (Driver)drivers.nextElement();
registeredDrivers.put(driver.getClass().getName(), driver);
}

}

在使用getConnection()方法以及所有重载时都会调用doGetConnection()方法

1
2
3
4
5
6
7
8
private Connection doGetConnection(Properties properties) throws SQLException {
// 初始化数据库驱动
this.initializeDriver();
Connection connection = DriverManager.getConnection(this.url, properties);
// 配置数据库连接的autoCommit和隔离级别
this.configureConnection(connection);
return connection;
}

initializeDriver()负责数据库驱动的初始化,会创建指定的Driver驱动对象,并将其注册到DriverManager中以及registeredDrivers中。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
private synchronized void initializeDriver() throws SQLException {
// 驱动是否已经注册
if (!registeredDrivers.containsKey(this.driver)) {
try {
Class driverType;
if (this.driverClassLoader != null) {
// 这个大家应该都很熟悉 Class.forName(driverName) 之前jdbc都是这么写的
driverType = Class.forName(this.driver, true, this.driverClassLoader);
} else {
driverType = Resources.classForName(this.driver);
}

Driver driverInstance = (Driver)driverType.getDeclaredConstructor().newInstance();
//注册驱动 UnpooledDataSource.DriverProxy是内部类,Driver的静态代理类
DriverManager.registerDriver(new UnpooledDataSource.DriverProxy(driverInstance));
registeredDrivers.put(this.driver, driverInstance);
} catch (Exception var3) {
throw new SQLException("Error setting driver on UnpooledDataSource. Cause: " + var3);
}
}

}

注册驱动这一步可能大家都不知道为什么会有这么一步,因为之前使用JDBC连接的时候只使用过DriverManager.getConnection获取连接,没有使用过DriverManager.registerDriver,而且可以正常使用驱动连接数据库获取数据,在这里给大家普及一下数据库的驱动

以Mysql驱动为例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
package com.mysql.jdbc;

import java.sql.DriverManager;
import java.sql.SQLException;

public class Driver extends NonRegisteringDriver implements java.sql.Driver {
public Driver() throws SQLException {
}

static {
try {
// 看到这个驱动类里的代码大家是不是就明白为什么之前不需要去手动注册驱动了吧
DriverManager.registerDriver(new Driver());
} catch (SQLException var1) {
throw new RuntimeException("Can't register driver!");
}
}
}

如果还是没明白,就看下述代码

1
2
3
4
5
// 这行代码就会去加载com.mysql.jdbc.Driver,这个类中的静态块在类加载时会执行,就会注册到DriverManager中
Class.forName("com.mysql.jdbc.Driver");

Connection connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/jdbc_demo","root","root");

PooledDataSource

数据库连接的创建是非常耗时的,数据库能够建立的连接数也是有限的,向上述UnpooledDataSource中这种每次都会创建连接的方式是存在弊端的,所以很多系统都是使用数据库连接池。

数据库连接池的好处

  • 可以实现数据库连接的重用

  • 提高响应速度

  • 防止数据库连接过多造成数据库假死

  • 避免数据库连接泄露

数据库连接池

数据库连接池在初始化时,会创建一定数量的数据库连接并添加到连接池中备用。当程序需要使用数据库连接时,从池中请求连接;当程序不再使用该连接时,会将其返回到池中缓存,等待下次使用,而不是直接关闭。数据库连接池可以设置连接总数上限以及空闲连接数上限,如果连接池创建的总连接数已达到上限,且都已被占用,则后续请求连接的线程会进入阻塞队列等待,直到有线程释放可用连接。如果连接池中空闲连接数较多,达到其上限,则后续返回的空闲连接不会放到池中,而是直接关闭,这样可以减少系统维护多余数据库连接的开销。

PooledDataSource

在使用PooledDataSource.getConnection的时候会调用popConnection方法,

1
2
3
public Connection getConnection(String username, String password) throws SQLException {
return this.popConnection(username, password).getProxyConnection();
}

在popConnection中操作的是PooledConnection,PooledConnection中封装了真正的数据库连接对象以及其代理对象(使用JDK动态代理产生的)

在PooledDataSource中有一个特别重要的属性

1
private final PoolState state = new PoolState(this);

PoolState是用来管理PooledConnection对象状态的组件,使用idleConnections和activeConnections两个集合来管理连接

1
2
3
4
// 空闲状态的连接
protected final List<PooledConnection> idleConnections = new ArrayList();
// 活跃状态的连接
protected final List<PooledConnection> activeConnections = new ArrayList();

PoolState还有一些是定义了一系列用来统计的字段

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
// 请求数据库次数
protected long requestCount = 0L;
// 获取连接的累积时间
protected long accumulatedRequestTime = 0L;
// checkoutTime表示从取出连接到归还连接的时长
// 所有连接的累积的checkoutTime时长
protected long accumulatedCheckoutTime = 0L;
// 超时的连接个数
protected long claimedOverdueConnectionCount = 0L;
// 累积超时时间
protected long accumulatedCheckoutTimeOfOverdueConnections = 0L;
// 累积等待时间
protected long accumulatedWaitTime = 0L;
// 等待次数
protected long hadToWaitCount = 0L;
// 无效连接数
protected long badConnectionCount = 0L;

PooledConnection是实现了InvocationHandler接口,

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
String methodName = method.getName();
//对close对象进行代理,调用pushConnection方法,会放回到连接池中,而不是真正的关闭
if ("close".equals(methodName)) {
this.dataSource.pushConnection(this);
return null;
} else {
try {
if (!Object.class.equals(method.getDeclaringClass())) {
this.checkConnection();
}

return method.invoke(this.realConnection, args);
} catch (Throwable var6) {
throw ExceptionUtil.unwrapThrowable(var6);
}
}
}