/**
* 功能1:从数据源获取数据库连接
* 功能2:从数据库获取到数据库连接后,绑定到本地线程(借助 ThreadLocal)
* 功能3:释放线程时和本地线程解除绑定
*/
public class JDBCUtils {
// 数据源成员变量设置为静态资源,保证大对象的单例性;同时保证静态方法中可以访问
private static DataSource dataSource;
// 由于 ThreadLocal 对象需要作为绑定数据时 k-v 对中的 key,所以要保证唯一性
// 加 static 声明为静态资源即可保证唯一性
private static ThreadLocal<Connection> threadLocal = new ThreadLocal<>();
// 在静态代码块中初始化数据源
static {
try {
// 操作思路分析:
// 从 jdbc.properties 文件中读取连接数据库的信息
// 为了保证程序代码的可移植性,需要基于一个确定的基准来读取这个文件
// 确定的基准:类路径的根目录。resources 目录下的内容经过构建操作中的打包操作后会确定放在 WEB-INF/classes 目录下。
// WEB-INF/classes 目录存放编译好的 *.class 字节码文件,所以这个目录我们就称之为类路径。
// 类路径无论在本地运行还是在服务器端运行都是一个确定的基准。
// 操作具体代码:
// 1、获取当前类的类加载器
ClassLoader classLoader = JDBCUtils.class.getClassLoader();
// 2、通过类加载器对象从类路径根目录下读取文件
InputStream stream = classLoader.getResourceAsStream("jdbc.properties");
// 3、使用 Properties 类封装属性文件中的数据
Properties properties = new Properties();
properties.load(stream);
// 4、根据 Properties 对象(已封装了数据库连接信息)来创建数据源对象
dataSource = DruidDataSourceFactory.createDataSource(properties);
} catch (Exception e) {
e.printStackTrace();
// 为了避免在真正抛出异常后,catch 块捕获到异常从而掩盖问题,
// 这里将所捕获到的异常封装为运行时异常继续抛出
throw new RuntimeException(e);
}
}
/**
* 工具方法:获取数据库连接并返回
* @return
*/
public static Connection getConnection() {
Connection connection = null;
try {
// 1、尝试从当前线程检查是否存在已经绑定的 Connection 对象
connection = threadLocal.get();
// 2、检查 Connection 对象是否为 null
if (connection == null) {
// 3、如果为 null,则从数据源获取数据库连接
connection = dataSource.getConnection();
// 4、获取到数据库连接后绑定到当前线程
threadLocal.set(connection);
}
} catch (SQLException e) {
e.printStackTrace();
// 为了调用工具方法方便,编译时异常不往外抛
// 为了不掩盖问题,捕获到的编译时异常封装为运行时异常抛出
throw new RuntimeException(e);
}
return connection;
}
/**
* 释放数据库连接
*/
public static void releaseConnection(Connection connection) {
if (connection != null) {
try {
// 在数据库连接池中将当前连接对象标记为空闲
connection.close();
// 将当前数据库连接从当前线程上移除
threadLocal.remove();
} catch (SQLException e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
}
}
JDBCUtils完整代码
最新推荐文章于 2023-09-18 13:51:47 发布