手工模拟数据库连接池

来源:互联网 发布:设置无线ap的软件 编辑:程序博客网 时间:2024/04/28 15:10
  1. /**********【目标】*************************************************************/  
  2. //该实例讲解了用Java如何编写实现一个数据库连接池  
  3. //通过编写数据库连接池,了解连接池内部运行机制和核心代码的实现  
  4. /**********【需解决的核心问题】**************************************************/  
  5. //首先,需要在程序中实例一个对象容器,保存建立好的数据库连接对象  
  6. //通过它对所有数据连接统一管理,获取连接池中连接的方法应该使用设计模式中的单子模式  
  7. //让所有想获得连接池中连接的外部程序只能通过同一个连接池管理对象来获得  
  8. //最好还有相应的处理机制,能够把超过一定时限没有使用的数据库连接对象强行收回  
  9. //同时能够调整连接池中的连接数,也就是可以设定连接池的初始大小  
  10. /**********【程序待改进的地方】**************************************************/  
  11. //这里编写的连接池程序功能上还很简单  
  12. //在实际中使用数据库连接池,不仅需要缓冲连接对象,通常还有缓冲Statement对象  
  13. //因为每申请一个连接(Connection)会在物理网络上建立一个用于通讯的连接  
  14. //在此连接上还可以申请一定数量的Statement  
  15. //一般的应用中,有多少个程序调用,则会产生多少次物理连接  
  16. //每个Statement单独占用一个物理连接,这是极大的资源浪费  
  17. //我们可以通过资源池实现对Statement对象的缓冲,让Statement对象也资源池化  
  18. //这样就可以几十、几百个Statement占占用一个物理连接,发挥数据库的最大性能  
  19. /**********【步骤】*************************************************************/  
  20. //1..本例中使用的数据库是Oracle10g_10.2.0.1.0数据库,用到的JDBC驱动为ojdbc6.jar  
  21. //   建表脚本如下:  
  22. //   create table guestbook(id number primary key, name varchar2(20) not null);  
  23. //2..在项目的根目录下新建一个dbpool.properties文件,这个文件是一个文本文件  
  24. //   记录了建立数据库连接池所需要的数据库参数,所以称该文件为连接池的配置文件  
  25. //   如果需要改变数据库的类型,只需要修改该文件中的内容,不需要修改连接池程序代码  
  26. //   dbpool.properties内容如下:  
  27. //   driverClassName=oracle.jdbc.driver.OracleDriver  
  28. //   username=scott  
  29. //   password=oracle  
  30. //   url=jdbc:oracle:thin:@127.0.0.1:1521:ora10  
  31. //   poolSize=10  
  32. //   下面简要说明dbpool.properties文件中的几个参数含义:  
  33. //   driverClassName----------代表数据库的驱动类  
  34. //   username和password-------代表数据库的用户名和密码  
  35. //   url----------------------代表建立数据库连接所需要的URL  
  36. //   poolSize-----------------代表连接池创建的初始连接数  
  37. //3..ConnectionPool.java------连接池的核心类  
  38. //   ConnectionPoolTest.java--连接池的测试类  
  39. //4..下面会附上ConnectionPool.java和ConnectionPoolTest.java的具体代码  
  40. /******************************************************************************/  
[java] view plaincopyprint?
  1. package com.jadyer.dbpool;  
  2.   
  3. import java.io.FileInputStream;  
  4. import java.sql.Connection;  
  5. import java.sql.SQLException;  
  6. import java.util.Properties;  
  7. import java.util.Vector;  
  8.   
  9. /** 
  10.  * 实现连接池的核心类 
  11.  * @author Administrator 
  12.  * 
  13.  */  
  14. public class ConnectionPool {  
  15.     private String url;  
  16.     private String username;  
  17.     private String password;  
  18.     private String driverClassName;  
  19.       
  20.     private Vector<Connection> pool; //用Vector类型的对象pool作为容器保存所有的数据库连接  
  21.     private int poolSize = 1//连接池的大小,也就是连接池中有多少个数据库连接  
  22.       
  23.     private static ConnectionPool instance = null//代表当前类的一个对象,它是静态的  
  24.   
  25.     /** 
  26.      * 私有的构造方法,禁止外部创建本类的对象 
  27.      * @see 要想获得该类的对象,可以通过<code>getIstance()</code>方法 
  28.      * @see 使用了单子模式,保证只有一个ConnectionPool对象供使用 
  29.      */  
  30.     private ConnectionPool() {  
  31.         init();  
  32.     }  
  33.   
  34.     /** 
  35.      * 连接池初始化方法,读取属性文件的内容,建立连接池中的初始连接 
  36.      */  
  37.     private void init() {  
  38.         pool = new Vector<Connection>(poolSize);  
  39.         readConfig(); //读取配置文件  
  40.         addConnection(); //向连接池中添加指定数目的连接  
  41.     }  
  42.       
  43.     /** 
  44.      * 返回连接到连接池中 
  45.      * @see 该方法让使用连接池的程序释放所使用的数据库连接 
  46.      * @see 并把被释放的连接重新添加到连接池中,供重复使用 
  47.      */  
  48.     public synchronized void release(Connection conn) {  
  49.         pool.add(conn);  
  50.     }  
  51.   
  52.     /** 
  53.      * 关闭连接池中的所有数据库连接 
  54.      * @see 当结束使用使用连接池的时候,可以调用该方法 
  55.      */  
  56.     public synchronized void closePool() {  
  57.         for (int i = 0; i < pool.size(); i++) {  
  58.             try {  
  59.                 //先获取每一个Connection对象,然后调用它们的close()方法,将其关闭  
  60.                 ((Connection) pool.get(i)).close();  
  61.             } catch (SQLException e) {  
  62.                 e.printStackTrace();  
  63.             }  
  64.             pool.remove(i); //从Vector中将其移除掉  
  65.         }  
  66.     }  
  67.   
  68.     /** 
  69.      * 返回当前连接池的一个对象 
  70.      */  
  71.     public static ConnectionPool getInstance() {  
  72.         if (instance == null) {  
  73.             instance = new ConnectionPool();  
  74.         }  
  75.         return instance;  
  76.     }  
  77.   
  78.     /** 
  79.      * 返回连接池中的一个数据库连接 
  80.      */  
  81.     public synchronized Connection getConnection() {  
  82.         if (pool.size() > 0) {  
  83.             Connection conn = pool.get(0); //从Vector中获取,获取的都是当前的第一个  
  84.             pool.remove(conn); //获取完之后,就从当前连接池中临时的移除掉  
  85.             return conn;  
  86.         } else {  
  87.             return null;  
  88.         }  
  89.     }  
  90.   
  91.     /** 
  92.      * 在连接池中创建初始设置的的数据库连接 
  93.      * @see 即建立数据库连接并且加入到pool对象中 
  94.      */  
  95.     private void addConnection() {  
  96.         Connection conn = null;  
  97.         for (int i = 0; i < poolSize; i++) {  
  98.             try {  
  99.                 Class.forName(driverClassName);  
  100.                 conn = java.sql.DriverManager.getConnection(url, username, password);  
  101.                 pool.add(conn);  
  102.             } catch (ClassNotFoundException e) {  
  103.                 e.printStackTrace();  
  104.             } catch (SQLException e) {  
  105.                 e.printStackTrace();  
  106.             }  
  107.   
  108.         }  
  109.     }  
  110.   
  111.     /** 
  112.      * 读取设置连接池的属性文件 
  113.      */  
  114.     private void readConfig() {  
  115.         try {  
  116.             //获取配置文件的路径  
  117.             String path = System.getProperty("user.dir") + "//dbpool.properties";  
  118.               
  119.             //通过IO的方式加载配置文件  
  120.             FileInputStream is = new FileInputStream(path);  
  121.             Properties props = new Properties();  
  122.             props.load(is);  
  123.               
  124.             //根据参数名读取指定参数的值  
  125.             this.driverClassName = props.getProperty("driverClassName");  
  126.             this.username = props.getProperty("username");  
  127.             this.password = props.getProperty("password");  
  128.             this.url = props.getProperty("url");  
  129.             this.poolSize = Integer.parseInt(props.getProperty("poolSize"));  
  130.         } catch (Exception e) {  
  131.             e.printStackTrace();  
  132.             System.err.println("读取属性文件出错..");  
  133.         }  
  134.     }  
  135. }  
[java] view plaincopyprint?
  1. package com.jadyer.dbpool;  
  2.   
  3. import java.sql.Connection;  
  4. import java.sql.DriverManager;  
  5. import java.sql.ResultSet;  
  6. import java.sql.Statement;  
  7.   
  8. /** 
  9.  * 连接池的测试类 
  10.  * @author Administrator 
  11.  * 
  12.  */  
  13. public class ConnectionPoolTest {  
  14.   
  15.     //分别计算使用连接池获得连接和传统方式获得连接所花费的时间  
  16.     //为了让对比更加明显,我们这里计算的是循环100次所花费的时间  
  17.     public static void main(String[] args) throws Exception {  
  18.         String sql = "select id,name from guestbook";  
  19.         long start = System.currentTimeMillis(); //获取当前的毫秒数  
  20.         ConnectionPool pool = null;  
  21.           
  22.         for (int i = 0; i < 100; i++) {  
  23.             pool = ConnectionPool.getInstance(); //获得一个连接池对象  
  24.             Connection conn = pool.getConnection(); //向连接池发出请求,获得一个连接对象  
  25.               
  26.             Statement stmt = conn.createStatement(); //获得到连接对象之后,用法就几乎一样了  
  27.             ResultSet rs = stmt.executeQuery(sql);  
  28.               
  29.             while (rs.next()) {}  
  30.               
  31.             rs.close();  
  32.             stmt.close();  
  33.             pool.release(conn); //将连接对象返回给连接池中,供重复使用  
  34.         }  
  35.           
  36.         pool.closePool(); //使用完连接池对象后,将所有的连接池对象释放掉  
  37.           
  38.         System.out.println("经过100次的循环调用,使用连接池花费的时间:" + (System.currentTimeMillis() - start) + "ms");  
  39.   
  40.           
  41.         /** 
  42.          * ************************************************ 
  43.          * 【上面为使用连接池的代码,下面为不使用连接池的代码】 
  44.          * ************************************************ 
  45.          */  
  46.   
  47.         //这里亦可使用String driverClass = "oracle.jdbc.driver.OracleDriver";  
  48.         //二者的区别是oracle.jdbc.OracleDriver继承了oracle.jdbc.driver.OracleDriver  
  49.         String driverClass = "oracle.jdbc.OracleDriver";  
  50.         String url = "jdbc:oracle:thin:@127.0.0.1:1521:ora10";  
  51.         String user = "scott";  
  52.         String password = "oracle";  
  53.         start = System.currentTimeMillis();  
  54.         for (int i = 0; i < 100; i++) {  
  55.             Class.forName(driverClass);  
  56.             Connection conn = DriverManager.getConnection(url, user, password);  
  57.             Statement stmt = conn.createStatement();  
  58.             ResultSet rs = stmt.executeQuery(sql);  
  59.             while (rs.next()) {  
  60.             }  
  61.             rs.close();  
  62.             stmt.close();  
  63.             conn.close();  
  64.         }  
  65.         System.out.println("经过100次的循环调用,未使用连接池花费时间:" + (System.currentTimeMillis() - start) + "ms");  
  66.     }  
  67. }  
[java] view plaincopyprint?
  1. /****【下面为ConnectionPoolTest.java的运行结果】****/  
  2. //经过100次的循环调用,使用连接池花费的时间:1125ms  
  3. //经过100次的循环调用,未使用连接池花费时间:2453ms  
  4. /**************************************************/  
原创粉丝点击