• 论题:

    • java c3p0获取连接Connnection 之后, 调用 con.close( ) 是否真的关闭了物理连接 ?
  • 简答:
    • c3p0采用连接池, 目的就是提前预置一定数量的连接, 在使用时候重复利用这些连接, 所以, con.close() 之后, 物理连接当然是被放回了连接池, 而没有真正的关闭 .
    • c3p0中的池连接(指:Connection) 是实现了PooledConnection接口的, PooledConnection为连接池管理提供钩子 (hook) 的对象, 它会在connnection被操作时接收到消息, 从而对该操作加以干涉,将Connection连接放回连接池.
  • 疑问:

    • 一些童鞋经过试验,发现如下问题,于是发出路上疑问, con.close()真的关闭了连接吗...
    •  public class MyTest {
      /**
      * 开启了20个线程,返回的hashcode,如下:没有重复的
      *1730967171 1122742512 156948859 1210106945 1217158489
      *1868341939 1595154849 1873764403 1937202425 1629470861
      */
      public static void main(String[] args) {
      for (int i = 0; i < 10; i++) {
      new Thread(new Runnable() {
      @Override
      public void run() { Connection con = JDBCUtil.getConnection();
      System.out.println(con.hashCode());
      // ....
      con.close();
      } catch (SQLException e) {
      e.printStackTrace();
      }
      }
      }).start();
      }
      }
      }
      public class JDBCUtil {
      /**
      * <!-- 最大空闲时间设置为0 , 即永不过期 -->
      * <property name="maxIdleTime">0</property>
      * <!-- 最多有多少个连接 -->
      * <property name="maxPoolSize">5</property>
      * <!-- 最少几个连接 -->
      * <property name="minPoolSize">2</property>
      */
      private static Connection conn;
      private static ComboPooledDataSource ds = new ComboPooledDataSource(); public static Connection getConnection() {
      try {
      conn = ds.getConnection();
      } catch (SQLException e) {
      e.printStackTrace();
      }
      return conn;
      }
      }
    • c3p0源码跟踪 [ps : 水平有限,以下未解释java事件注册相关原理 ]

    •  abstract class AbstractPoolBackedDataSource
      extends PoolBackedDataSourceBase
      implements PooledDataSource{
      //..... // dataSource.getConnection()所调用的就是该方法
      //implementation of javax.sql.DataSource
      public Connection getConnection() throws SQLException{ PooledConnection pc =
      getPoolManager(). // 返回 C3P0PooledConnectionPoolManager
      getPool(). // 返回 C3P0PooledConnectionPool
      checkoutPooledConnection(); // 返回PooledConnection
      return pc.getConnection();
      } //.....
      } class C3P0PooledConnectionPool{
      // ....
      public PooledConnection checkoutPooledConnection() throws SQLException{
      //System.err.println(this + " -- CHECKOUT");
      try
      {
      PooledConnection pc = (PooledConnection) this.checkoutAndMarkConnectionInUse();
      pc.addConnectionEventListener( cl );
      return pc;
      }
      catch (TimeoutException e)
      { throw SqlUtils.toSQLException("An attempt by a client to checkout a Connection has timed out.", e); }
      catch (CannotAcquireResourceException e)
      { throw SqlUtils.toSQLException("Connections could not be acquired from the underlying database!", "08001", e); }
      catch (Exception e)
      { throw SqlUtils.toSQLException(e); }
      } private Object checkoutAndMarkConnectionInUse() throws TimeoutException, CannotAcquireResourceException, ResourcePoolException, InterruptedException
      {
      Object out = null;
      boolean success = false;
      while (! success)
      {
      try
      {
      out = rp.checkoutResource( checkoutTimeout );
      if (out instanceof AbstractC3P0PooledConnection)
      {
      // cast should succeed, because effectiveStatementCache implies c3p0 pooled Connections
      AbstractC3P0PooledConnection acpc = (AbstractC3P0PooledConnection) out;
      /*
      *以下在获取物理连接的时候,PooledcConnection中注册的事件监听器会收到消息
      *无论每次对connection的何种操作,PooledConnection都会收到来自驱动的消息,
      *其中的钩子(hook)对象就会完成对Connection的回收
      */
      Connection physicalConnection = acpc.getPhysicalConnection();
      success = tryMarkPhysicalConnectionInUse(physicalConnection);
      }
      else
      success = true; //we don't pool statements from non-c3p0 PooledConnections
      }
      finally
      {
      try { if (!success && out != null) rp.checkinResource( out );}
      catch (Exception e) { logger.log(MLevel.WARNING, "Failed to check in a Connection that was unusable due to pending Statement closes.", e); }
      }
      }
      return out;
      }
      //.... }
  • 综述:
    • 当应用程序关闭连接时,它调用 Connection 方法 close。完成连接池时,连接池管理器将得到通知;因为它曾使用 ConnectionPool 方法 addConnectionEventListener 作为 ConnectionEventListener 对象注册它自身。连接池管理器释放到 PooledConnection 对象的句柄,并将 PooledConnection 对象返回到连接池,以便再次使用。因此,当应用程序关闭其连接时,基础物理连接会被回收而不是被关闭。
    • 知识水平有限,难免错误,模糊, 请多批评指正,谢谢.

c3p0获取连接Connection后的Close()---释疑的更多相关文章

  1. mysql获取连接connection失败

    好久不写jdbc了,今天写了个小东西,数据库连接失败,错误信息如下: java.sql.SQLException: The server time zone value '???ú±ê×??±??' ...

  2. 选择c3p0作为连接池

    <hibernate-configuration>     <session-factory>         <property name="dialect& ...

  3. Hibernate整合C3P0实现连接池

    Hibernate整合C3P0实现连接池 hibernate中可以使用默认的连接池,无论功能与性能都不如C3PO(网友反映,我没有测试过),C3P0是一个开源的JDBC连接池,它实现了数据源和JNDI ...

  4. JDBC获取数据库Connection的工具抽取

    使用JDBC获取数据库的连接,大字分为三个步骤 1.获取驱动包名,定义URL,database_username,database_password 2.获取Connection对象 3.利用Conn ...

  5. 事务处理中如何获取同一个connection 对象

    运用线程内部的map属性,将对象绑定到ThreadLocal中: 具体实现: 1.新建一个绑定Connection对象的单例类 public class ConnectionBind { privat ...

  6. JDBC(二)—— 获取连接池方式

    ## 获取数据库连接的方式 ### 方式一 ```javaDriver driver = new com.mysql.cj.jdbc.Driver(); String url = "jdbc ...

  7. WzwJDBC 自定义工具类(获取连接,释放资源)

    package wzwUtil;import java.io.IOException;import java.io.InputStream;import java.sql.*;import java. ...

  8. idea启动服务连接mysql后 Navicat连接mysql就报错2013-Lost connection toMySQL server at

    我是使用navicat的windows端 连接centos下mysql服务器 第一次常规连接mysql正常,idea启动服务连接mysql后 Navicat连接mysql就报错2013-Lost co ...

  9. Java Web(十) JDBC的增删改查,C3P0等连接池,dbutils框架的使用

    前面做了一个非常垃圾的小demo,真的无法直面它,菜的抠脚啊,真的菜,好好努力把.菜鸡. --WH 一.JDBC是什么? Java Data Base Connectivity,java数据库连接,在 ...

随机推荐

  1. linux 下source、sh、bash、./执行脚本的区别

    原文地址:http://blog.csdn.net/caesarzou/article/details/7310201 source命令用法: source FileName 作用:在当前bash环境 ...

  2. OC纯代码全手工打造ScroolView实现翻页

    OC纯代码全手工打造ScroolView实现翻页 1. 概述 分为三部分: 上部标题ScrollView 下部内容ScrollView 上部当前页 标示线 2. 效果 上下两部分都随着手势的滑动一块滑 ...

  3. 那就用pthon来写个跳板机吧

    1.需求 程序一: 1.后台管理 - 堡垒机上创建用户和密码(堡垒机root封装的类,UserProfile表) - .bashrc /usr/bin/python3 /data/bastion.py ...

  4. iOS 之 Strong与Weak,_unsafe _unretained与weak区别

    1. 在ARC中 strong(强引用) 相当于retain, weak(弱引用) 相当于assign.ARC下,strong告诉编译器自动插入retain.但是在ARC下,代理协议的属性依然用ass ...

  5. 浅谈JavaWEB入门必备知识之Servlet入门案例详解

    工欲善其事.必先利其器,想要成为JavaWEB高手那么你不知道servlet是一个什么玩意的话,那就肯定没法玩下去,那么servlet究竟是个什么玩意?下面,仅此个人观点并通过一个小小的案例来为大家详 ...

  6. spring mvc在Controller中获取ApplicationContext

    spring mvc在Controller中获取ApplicationContext web.xml中进行正常的beans.xml和spring-mvc.xml的配置: 需要在beans.xml中进行 ...

  7. document.body.clientHeight的取值

    http://www.cnblogs.com/fullhouse/archive/2012/01/05/2313800.html 有时候需要取页面的底部, 就会用到document.body.clie ...

  8. 关于GCD同步组实现多个异步线程的同步执行中的注意点

    在App开发中经常会遇到多个线程同时向服务器取数据, 如果每个线程取得数据后都去刷新UI会造成界面的闪烁 也有可能出现部分数据还没有获取完毕造成程序crash 之前在网上看到很多是利用dispatch ...

  9. 锁 和 CopyOnWrite的实现

    1.普通锁 只有lock功能, Java实现:ReentrantLock lock = new ReentrantLock(); lock和unlock: lock.lock(); lock.unlo ...

  10. HDU2063(二分图最大匹配)

    过山车 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submiss ...