JDBC连接数据库之连接池(5)

来源:互联网 发布:淘宝十大黑科技 编辑:程序博客网 时间:2024/05/22 09:49

传统的模式是:

-在主程序中(servlet,beans中)建立数据库连接

-进行sql操作,

-断开数据库连接

这种模式开发存在的问题:

-普通的JDBC数据库使用DriverManager来获取,每次向数据库建立连接的时候都要将Connection加载到内存中在验证用户名和密码(得花费0.05s~1s的时间)。

需要数据库连接的时候,就像数据库要求一个,执行完成后在断开连接。这样的方式将会消耗大量的资源和时间。数据库的连接资源并没有得到很好的重复利用。

若同时有近百人甚至几千人在线,平凡的进行数据库连接操作将占用很多的系统资源,严重的甚至造成服务器崩溃。

-对于每一次数据库连接,使用完后都得断开。否则,如果程序出现异常而未能关闭将会导致数据库系统中的内存泄漏,最终导致重启数据库。

-这种开发不能控制被创建的连接对象数,系统资源会被毫无顾及的分配出去,如连接过多,也可能导致内存泄漏,服务器崩溃。

数据库连接池的基本思想是:

未解决传统开发的数据库连接问题,可以采用数据库连接池技术。

数据库连接池的基本思想就是为数据库建立一个”缓冲池“。预先在缓冲池中放入一定数量的连接,当需要建立数据库连接时,只需从“缓冲池”中取出一个,使用完毕

之后再放回去。

数据库连接池负责分配,管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是重新建一个。

数据库连接池在初始化时将创建一定数量的数据库连接放到连接池中,这些数据库连接的数量是由最小数据库连接数来设定的。无论这些数据库连接是否被使用,连接

池都将保证至少拥有这么多的连接数量。连接池的最大数据库连接数量限定了这个连接池能战友的最大连接数,当应用程序向连接池能占有的最大连接数,当应用程序

向连接池请求的连接数朝富哦最大连接数量时,这些请求将被加入到等待队列中。



下面介绍两种开源的数据库连接池。

JDBC的数据库连接池使用javax.sql.DataSource来表示,DataSource只是一个接口,该接口通常有服务器(Weblogic,WebSphere,Tomcat)提供实现

也有一些开源实现:

——DBCP数据库连接池

——C3P0数据库连接池

DataSource通常被称为数据源,它包含连接池和连接池管理两个部分,习惯上也把DataSource成为连接池。

————————DBCP连接池————————

apatch提供的开源jar,也是tomcat内置连接池。(一家的)

jar:commons-dbcp.jar

依赖的jar包 commons

//使用DBCP数据库连接池public void testJDBC() throws SQLException{final BasicDataSource dataSource=new BasicDataSource();//1创建DBCP数据源实例//2为数据源实例指定必须的属性dataSource.setUsername("root");dataSource.setPassword("root");dataSource.setUrl("jdbc:mysql:///test");dataSource.setDriverClassName("com.mysql.jdbc.Driver");//3指定数据源的一些可选属性。//1)指定数据库连接池中初始化连接数的个数dataSource.setInitialSize(5);         //2)指定最大的连接数(同一时刻可以同时向数据库申请的连接数。)dataSource.setMaxActive(5);//3)指定最小连接数:在数据库连接池中保存的最少的空闲的连接的数量。dataSource.setMinIdle(2);//4)等待数据库连接池分配连接的最长时间。单位为毫秒。超出改时间抛出异常。dataSource.setMaxWait(1000*5);//从数据源中获取数据库连Connection connection=dataSource.getConnection();System.out.println(connection.getClass()); connection=dataSource.getConnection();System.out.println(connection.getClass()); connection=dataSource.getConnection();System.out.println(connection.getClass()); connection=dataSource.getConnection();System.out.println(connection.getClass());Connection connection2=dataSource.getConnection();System.out.println(" > "+connection2.getClass());      new Thread(){           public void run() {      Connection conn;      try{    conn=dataSource.getConnection();    System.out.println(conn.getClass());      }catch(SQLException e){      e.printStackTrace();      }           };      }.start();      try{      Thread.sleep(3000);      }catch(InterruptedException e){      e.printStackTrace();      }      connection2.close();}
使用BasicDataSourceFactory

步骤:

1.加载cbcp的properties配置文件:配置文件中的键需要来自BasicDataSource的属性

2.调用BasicDataSourceFactory的createDataSource方法创建DataSource实例

3.从DataSource实例中获取数据库连接。

//使用DBCP创建连接。/* * 1加入jar包(两个jar)依赖于commons Poos * 2创建数据库连接池 * 3为数据据实例指定必须的属性。 * 4从数据中获取数据库连接。*/public void testDBCPWithDataSourceFactory() throws Exception{Properties properties=new Properties();InputStream inStream=JDBCTest.class.getClassLoader().getResourceAsStream("dbcp.properties");properties.load(inStream);DataSource dataSource=BasicDataSourceFactory.createDataSource(properties);  System.out.println(dataSource.getConnection());}
其中Properties文件dbcp.properties放在src目录下且内容为:

username=rootpassword=rootdriverClassName=com.mysql.jdbc.Driverurl=jdbc\:mysql\:///testinitialSize=10maxActive=50minIdle=5maxWait=5000

——————C3P0数据源——————

jar: C3P0.jar

//测试c3p0数据源。public void testC3P0() throws Exception{ComboPooledDataSource cpds = new ComboPooledDataSource();cpds.setDriverClass( "com.mysql.jdbc.Driver" ); //loads the jdbc driver            cpds.setJdbcUrl( "jdbc:mysql:///test" );cpds.setUser("root");                                  cpds.setPassword("root");   System.out.println(cpds.getConnection()); }
使用配置文件,获取数据源:

c3p0-config.xml文件内容为:

<c3p0-config>  <named-config name="helloc3p0">   <!-- 指定连接数据源的基本属性 -->    <property name="user">root</property>    <property name="password">root</property>    <property name="driverClass">com.mysql.jdbc.Driver</property>    <property name="jdbcUrl">jdbc:mysql:///test</property>     <!-- 若数据库中连接数不足时,一次向数据库服务器申请多少连接。 -->    <property name="acquireIncrement">50</property>      <!-- 初始化数据库连接池连接的数量 -->    <property name="initialPoolSize">5</property>      <!-- 数据库连接池中最小的数据库连接数 -->    <property name="minPoolSize">5</property>    <!-- 数据库连接池中最大的数据库连接数 -->    <property name="maxPoolSize">1000</property>    <!-- 数据库连接池可以维护的Statement的个数。 -->    <property name="maxStatements">20</property>      <!-- 每个连接同时可以使用的Statement对象的个数-->    <property name="maxStatementsPerConnection">5</property>  </named-config></c3p0-config>
/*1创建成p3p0-config.xml文件,参考帮助文档中 *创建ComboPooledDataSouce实例。 *从DataSource实例中获取数据库连接。 * */public void testC3p0WithConfigFile() throws SQLException{DataSource dataSource=new ComboPooledDataSource("helloc3p0");System.out.println(dataSource.getConnection());ComboPooledDataSource comboPooledDataSource=(ComboPooledDataSource)dataSource;System.out.println(comboPooledDataSource.getMaxStatements());}
修改JDBCTool为:

package com.atguigu.jdbc;import java.io.IOException;import java.io.InputStream;import java.sql.Connection;import java.sql.DriverManager;import java.sql.ResultSet;import java.sql.SQLException;import java.sql.Statement;import java.util.Properties;import javax.sql.DataSource;import com.mchange.v2.c3p0.ComboPooledDataSource;/** * JDBC 的工具类 *  * 其中包含: 获取数据库连接, 关闭数据库资源等方法. */public class JDBCTools {public static void commit(Connection connection){if(connection!=null){try{connection.commit();}catch(SQLException e){e.printStackTrace();}}}public static void rollback(Connection connection){if(connection!=null){try {connection.rollback();} catch (SQLException e) {e.printStackTrace();}}}public static void beginTx(Connection connection){if(connection!=null){try {connection.setAutoCommit(false);} catch (SQLException e) {e.printStackTrace();}}}//////////////////////////////数据库连接池应只被初始化一次private static DataSource dataSource=null;static{dataSource=new ComboPooledDataSource("helloc3p0");}public static Connection getConnection() throws Exception {/*Properties properties = new Properties();InputStream inStream = JDBCTools.class.getClassLoader().getResourceAsStream("jdbc.properties");properties.load(inStream);// 1. 准备获取连接的 4 个字符串: user, password, jdbcUrl, driverClassString user = properties.getProperty("user");String password = properties.getProperty("password");String jdbcUrl = properties.getProperty("jdbcUrl");String driverClass = properties.getProperty("driverClass");// 2. 加载驱动: Class.forName(driverClass)Class.forName(driverClass);// 3. 调用// DriverManager.getConnection(jdbcUrl, user, password)// 获取数据库连接Connection connection = DriverManager.getConnection(jdbcUrl, user,password);return connection;*/return dataSource.getConnection();}public static void releaseDB(ResultSet resultSet, Statement statement,Connection connection) {if (resultSet != null) {try {resultSet.close();} catch (SQLException e) {e.printStackTrace();}}if (statement != null) {try {statement.close();} catch (SQLException e) {e.printStackTrace();}}if (connection != null) {try {//数据库连接池的Connection对象进行close时//并不是真的进行关闭,而是把该数据库连接归还到数据库连接池中connection.close();} catch (SQLException e) {e.printStackTrace();}}}}






0 0
原创粉丝点击