package com.JDBC.firstJDBC;
import java.util.*;
import java.io.*;
import java.sql.*;
/*此类用于封装Java程序链接数据库所需要配置的JDBC操作
*
* */
public class SingletonJDBCUtil{
private static Properties info=new Properties();
private static Connection conn=null;
static{
try{
//用于读取配置文件中的字节流
//getResourceAsStream方法用于类加载器在加载的时候并加载制指定的资源。
InputStream is=JDBCUtil.class.getResourceAsStream("/com" +
"/JDBC/firstJDBC/config.properties");
info.load(is);
is.close();
}catch(Exception e){
//此处只能跑出此种异常,这种异常专门用来在初始化快中抛出的
throw new ExceptionInInitializerError(e);
}
}
//返回一个连接
public static Connection getConnection()throws Exception{
if(conn==null){
Class.forName(info.getProperty("driver"));
conn=DriverManager.getConnection(info.getProperty("url"),
info.getProperty("username"),info.getProperty("pwd"));
}
return conn;
}
//释放一个资源
public static void release(ResultSet rs,Statement stm,Connection cn){
if(rs!=null) try{rs.close();}catch(Exception e){}
if(stm!=null) try{stm.close();}catch(Exception e){}
if(cn!=null) try{cn.close();}catch(Exception e){}
}
}
但上述连接代码存在严重的资源浪费的情况,比如任何一个对象只要调用JDBCUtil的getConnection()方法,就会得到一个连接,这样大大的浪费了资源,以下通过singleton设计模式进行改进后的代码如下:
package com.JDBC.firstJDBC;
import java.util.*;
import java.io.*;
import java.sql.*;
/*此类用于封装Java程序链接数据库所需要配置的JDBC操作
*
* */
public class SingletonJDBCUtil{
private static Properties info=new Properties();
private static Connection conn=null;
static{
try{
//用于读取配置文件中的字节流
//getResourceAsStream方法用于类加载器在加载的时候并加载制指定的资源。
InputStream is=JDBCUtil.class.getResourceAsStream("/com" +
"/JDBC/firstJDBC/config.properties");
info.load(is);
is.close();
}catch(Exception e){
//此处只能跑出此种异常,这种异常专门用来在初始化快中抛出的
throw new ExceptionInInitializerError(e);
}
}
//返回一个连接
public static Connection getConnection()throws Exception{
if(conn==null){
Class.forName(info.getProperty("driver"));
conn=DriverManager.getConnection(info.getProperty("url"),
info.getProperty("username"),info.getProperty("pwd"));
}
return conn;
}
//释放一个资源
public static void release(ResultSet rs,Statement stm,Connection cn){
if(rs!=null) try{rs.close();}catch(Exception e){}
if(stm!=null) try{stm.close();}catch(Exception e){}
if(cn!=null) try{cn.close();}catch(Exception e){}
}
}
这种代码仍然存在问题,当多个线程同时调用该连接的时候,当某一个线程关闭该连接后,其他线程由于还有某些操作没有提交而导致业务的错误.因此,需要一个线程一个连接,代码如下:
package com.JDBC.firstJDBC;
import java.util.*;
import java.io.*;
import java.sql.*;
/*此类用于封装Java程序链接数据库所需要配置的JDBC操作
*
* */
public class BestUtil{
private static Properties info=new Properties();
static{
try{
//用于读取配置文件中的字节流
//getResourceAsStream方法用于类加载器在加载的时候并加载制指定的资源。
InputStream is=JDBCUtil.class.getResourceAsStream("/com" +
"/JDBC/firstJDBC/config.properties");
info.load(is);
is.close();
}catch(Exception e){
//此处只能跑出此种异常,这种异常专门用来在初始化快中抛出的
throw new ExceptionInInitializerError(e);
}
}
//返回一个连接
private static final ThreadLocal<Connection> tl=new ThreadLocal<Connection>();
public static Connection getConnection()throws Exception{
Connection conn=tl.get();
if(conn==null){
Class.forName(info.getProperty("driver"));
conn=DriverManager.getConnection(info.getProperty("url"),
info.getProperty("username"),info.getProperty("pwd"));
tl.set(conn);
}
return conn;
}
//释放一个资源
public static void release(ResultSet rs,Statement stm,Connection cn){
if(rs!=null) try{rs.close();}catch(Exception e){}
if(stm!=null) try{stm.close();}catch(Exception e){}
if(cn!=null) try{cn.close();}catch(Exception e){}
}
}
这种模式解决了在前面出现并发操作的问题,是一种你比较优秀的数据连接封装操作,其中用到了ThreadLocal类,每一个线程都有一个用于存储线程的局部变量(线程独占数据,用Map对象来存储数据):由于线程的局部变是属于java.lang包的 defalut类型,因此,要访问该对象就必须用调用ThreadLocal对象来获的里面的对象。
分享到:
相关推荐
javase集合 温故而知新.doc
CPU 技术温故而知新.pdf
CPU 技术温故而知新(之三).pdf
食品饮料行业深度报告:温故而知新,关注本轮大众品提价与盈利传导(2022)(23页).pdf
5.9 MemoryStream 简单示例 :自定义一个处理图片的HttpHandler 6.1 简单介绍一下BufferedStream 6.2 如何理解缓冲区? 6.3 BufferedStream的优势 6.4 从BufferedStream 中学习装饰模式 6.5 如何理解装饰模式 6.6 ...
很基础的电子基础东西,相信很多朋友是不需要的,不过还是放上去,有需要回顾一下的~~~~~~~ 请看这里!
dos系统概述及详解 DOS系统简介 dos系统是典型的什么系统 DOS基本概念 dos百度百科 dos系统使用 dos常用命令 dos系统基本命令
NULL 博文链接:https://wellfrog.iteye.com/blog/1133725
主要给大家介绍了关于JS温故而知新之变量提升和时间死区的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
主要介绍了Spring MVC温故而知新系列教程之请求映射RequestMapping注解的相关知识,文中给大家介绍了RequestMapping注解提供的几个属性及注解说明,感兴趣的朋友跟随脚本之家小编一起学习吧
闭包是将一些执行语句的封装,可以将封装的结果像对象一样传递,在传递时,这个封装依然能够访问到原上下文。下面这篇文章主要给大家介绍了关于C#中闭包的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以...
Spring MVC 框架在 Java 的 Web 项目中应该是无人不知的吧,你不会搭建一个 Spring 框架?作为身为一个刚刚学习Java的我都会,如果你不会的话,那可真令人忧伤。下面这篇文章主要给大家介绍了关于Spring MVC从零开始...
任何一个元素都可以指定为flexbox布局,其中设为display:flex或者display:inline-flex的元素称为伸缩容器。伸缩容器的子元素称为伸缩项目,伸缩项目使用伸缩布局模型来排版。伸缩布局模型与传统的布局不一样,它按照...
温故而知新
mtk实例教程(新手入门-老手温故) mtk实例教程(新手入门-老手温故)
leetcode 苹果 andy-tools 分享知识和分享苹果有着本质区别 分享知识可以从中得到快乐,然而分享苹果并不能, 工程目录 [结构模式(Structural)]