一、定义

C3P0是一个开源的JDBC连接池,它实现了数据源与JNDI绑定,支持JDBC3规范和实现了JDBC2的标准扩展说明的Connection和Statement池的DataSources对象。

  即将用于连接数据库的连接整合在一起形成一个随取随用的数据库连接池(Connection pool)。

二、为什么使用C3P0数据库连接池

2.1、时间和内存资源消耗巨大

  • 使用JDBC传统模式

public static void main(String[] args) {BaseDao2<Dog> dao = new BaseDao2<>();System.out.println(Timestamp.valueOf(LocalDateTime.now()));for(int i=0;i<100;i++){Connection connection=dao.getConnection();try {connection.close();} catch (SQLException e) {e.printStackTrace();}}System.out.println(Timestamp.valueOf(LocalDateTime.now()));}
  • 使用C3P0连接

public static void main(String[] args) {BaseDao2<Dog> dao = new BaseDao2<>();System.out.println(Timestamp.valueOf(LocalDateTime.now()));for(int i=0;i<100;i++){Connection connection = getConnection();try {connection.close();} catch (SQLException e) {e.printStackTrace();}}System.out.println(Timestamp.valueOf(LocalDateTime.now()));}

我们进行了100次的连接,使用JDBC传统模式连接,耗时近2s,而使用C3P0数据库连接池时,耗时1s不到的时间。当数据量大的时候,耗时的差距会更加明显。

会出现这种情况是因为,普通的JDBC数据库连接使用DriverManager来获取,每次向数据库建立连接的时候都要将Connection加载到内存中,再根据JDBC代码(或配置文件)中的用户名和密码进行验证其正确性。这一过程一般会花费0.05~1s,一旦需要数据库连接的时候就必须向数据库请求一个,执行完后再断开连接。

如果同一个数据库在同一时间有数十人甚至上百人请求连接势必会占用大量的系统资源,严重的会导致服务器崩溃。

2.2、有内存泄漏的风险

每一次数据库连接使用完后都需要断开连接,但如果程序出现异常致使连接未能及时关闭,这样就可能导致内存泄漏,最终只能以重启数据库的方法来解决;

另外使用传统JDBC模式开发不能控制需要创建的连接数,系统一般会将资源大量分出给连接以防止资源不够用,如果连接数超出一定数量也会有极大的可能导致内存泄漏。

2.3、应用程序直接获取链接

缺点:

  1. 用户每次请求都需要向数据库获得链接,而数据库创建连接通常需要消耗相对较大的资源,创建时间也较长

  1. 假设网站一天10万访问量,数据库服务器就需要创建10万次连接,极大的浪费数据库的资源,并且极易造成数据库服务器内存溢出、宕机

2.4、使用数据库连接池进行优化

三、C3P0实操

3.1、导入jar包

在pom文件中添加如下内容:

<dependency><groupId>com.mchange</groupId><artifactId>c3p0</artifactId><version>0.9.5.2</version>
</dependency>

3.2、配置xml文件

下面是配置的xml文件,可以根据实际进行修改。

3.3、c3p0-config.xml模板

<c3p0-config><named-config name="mysqlapp"><property name="driverClass">com.mysql.cj.jdbc.Driver</property><property name="jdbcUrl">jdbc:mysql://192.168.153.134:3306/new</property><property name="user">root</property><property name="password">123123</property><!-- 进行数据库连接池管理的基本信息 --><!-- 当数据库连接池中的连接数不够时,c3p0一次性向数据库服务器申请的连接数 --><property name="acquireIncrement">5</property><!-- c3p0数据库连接池中初始化时的连接数 --><property name="initialPoolSize">10</property><!-- c3p0数据库连接池维护的最少连接数 --><property name="minPoolSize">10</property><!-- c3p0数据库连接池维护的最多的连接数 --><property name="maxPoolSize">200</property><!-- c3p0数据库连接池最多维护的Statement的个数 --><property name="maxStatements">50</property><!-- 每个连接中可以最多使用的Statement的个数 --><property name="maxStatementsPerConnection">2</property></named-config>
</c3p0-config>

3.4、c3p0-config.xml参数清单

<c3p0-config>   <default-config>   <!--当连接池中的连接耗尽的时候c3p0一次同时获取的连接数。Default: 3 -->   <property name="acquireIncrement">3</property>   <!--定义在从数据库获取新连接失败后重复尝试的次数。Default: 30 -->   <property name="acquireRetryAttempts">30</property>   <!--两次连接中间隔时间,单位毫秒。Default: 1000 -->   <property name="acquireRetryDelay">1000</property>   <!--连接关闭时默认将所有未提交的操作回滚。Default: false -->   <property name="autoCommitOnClose">false</property>   <!--c3p0将建一张名为Test的空表,并使用其自带的查询语句进行测试。如果定义了这个参数那么   属性preferredTestQuery将被忽略。你不能在这张Test表上进行任何操作,它将只供c3p0测试   使用。Default: null-->   <property name="automaticTestTable">Test</property>   <!--获取连接失败将会引起所有等待连接池来获取连接的线程抛出异常。但是数据源仍有效   保留,并在下次调用getConnection()的时候继续尝试获取连接。如果设为true,那么在尝试   获取连接失败后该数据源将申明已断开并永久关闭。Default: false-->   <property name="breakAfterAcquireFailure">false</property>   <!--当连接池用完时客户端调用getConnection()后等待获取新连接的时间,超时后将抛出   SQLException,如设为0则无限期等待。单位毫秒。Default: 0 -->   <property name="checkoutTimeout">100</property>   <!--通过实现ConnectionTester或QueryConnectionTester的类来测试连接。类名需制定全路径。   Default: com.mchange.v2.c3p0.impl.DefaultConnectionTester-->   <property name="connectionTesterClassName"></property>   <!--指定c3p0 libraries的路径,如果(通常都是这样)在本地即可获得那么无需设置,默认null即可   Default: null-->   <property name="factoryClassLocation">null</property>   <!--强烈不建议使用该方法,将这个设置为true可能会导致一些微妙而奇怪的bug-->   <property name="forceIgnoreUnresolvedTransactions">false</property>   <!--每60秒检查所有连接池中的空闲连接。Default: 0 -->   <property name="idleConnectionTestPeriod">60</property>   <!--初始化时获取三个连接,取值应在minPoolSize与maxPoolSize之间。Default: 3 -->   <property name="initialPoolSize">3</property>   <!--最大空闲时间,60秒内未使用则连接被丢弃。若为0则永不丢弃。Default: 0 -->   <property name="maxIdleTime">60</property>   <!--连接池中保留的最大连接数。Default: 15 -->   <property name="maxPoolSize">15</property>   <!--JDBC的标准参数,用以控制数据源内加载的PreparedStatements数量。但由于预缓存的statements   属于单个connection而不是整个连接池。所以设置这个参数需要考虑到多方面的因素。   如果maxStatements与maxStatementsPerConnection均为0,则缓存被关闭。Default: 0-->   <property name="maxStatements">100</property>   <!--maxStatementsPerConnection定义了连接池内单个连接所拥有的最大缓存statements数。Default: 0 -->   <property name="maxStatementsPerConnection"></property>   <!--c3p0是异步操作的,缓慢的JDBC操作通过帮助进程完成。扩展这些操作可以有效的提升性能   通过多线程实现多个操作同时被执行。Default: 3-->   <property name="numHelperThreads">3</property>   <!--当用户调用getConnection()时使root用户成为去获取连接的用户。主要用于连接池连接非c3p0   的数据源时。Default: null-->   <property name="overrideDefaultUser">root</property>   <!--与overrideDefaultUser参数对应使用的一个参数。Default: null-->   <property name="overrideDefaultPassword">password</property>   <!--密码。Default: null-->   <property name="password"></property>   <!--定义所有连接测试都执行的测试语句。在使用连接测试的情况下这个一显著提高测试速度。注意:   测试的表必须在初始数据源的时候就存在。Default: null-->   <property name="preferredTestQuery">select id from test where id=1</property>   <!--用户修改系统配置参数执行前最多等待300秒。Default: 300 -->   <property name="propertyCycle">300</property>   <!--因性能消耗大请只在需要的时候使用它。如果设为true那么在每个connection提交的   时候都将校验其有效性。建议使用idleConnectionTestPeriod或automaticTestTable   等方法来提升连接测试的性能。Default: false -->   <property name="testConnectionOnCheckout">false</property>   <!--如果设为true那么在取得连接的同时将校验连接的有效性。Default: false -->   <property name="testConnectionOnCheckin">true</property>   <!--用户名。Default: null-->   <property name="user">root</property>   <!--早期的c3p0版本对JDBC接口采用动态反射代理。在早期版本用途广泛的情况下这个参数   允许用户恢复到动态反射代理以解决不稳定的故障。最新的非反射代理更快并且已经开始   广泛的被使用,所以这个参数未必有用。现在原先的动态反射与新的非反射代理同时受到   支持,但今后可能的版本可能不支持动态反射代理。Default: false-->   <property name="usesTraditionalReflectiveProxies">false</property> </default-config>
</c3p0-config>

3.5、创建C3P0Util类

// C3p0连接池
public class C3p0Util {private static DataSource dataSource = null;static{dataSource = new ComboPooledDataSource("mysqlapp");}//从连接池中获取连接public static Connection getConnection(){try {return dataSource.getConnection();} catch (SQLException e) {e.printStackTrace();        }return null;}//释放连接回连接池public static void release(Connection conn, Statement stmt, ResultSet rs) {if (rs != null) {try {rs.close();} catch (Exception e) {e.printStackTrace();}rs = null;}if (stmt != null) {try {stmt.close();} catch (Exception e) {e.printStackTrace();}stmt = null;}if (conn != null) {try {conn.close();} catch (Exception e) {e.printStackTrace();}conn = null;}}
}

C3P0(数据库连接池)详解相关推荐

  1. JDBC中C3PO数据库连接池详解

    -----------------------------------------------------JDBC中C3PO数据库连接池详解------------------------------ ...

  2. Spring Boot使用hikari、druid、c3p0等数据库连接池详解

    文章目录 前言 Hikari连接池 Druid连接池 Druid(新版starter)连接池 C3P0连接池(1 C3P0连接池(2 扩展 前言 截至Spring Boot V2.0为止,官方仅为下列 ...

  3. Druid 数据库连接池 详解

    推荐阅读:JDBC详解 文章目录 概述 数据库连接池实现 Driud使用流程 代码示例 概述 1.数据库连接池是个容器,负责分配.管理数据库连接(Connection): 2.它允许应用程序重复使用一 ...

  4. Java数据库开发与应用之MySQL数据库、JDBC操作数据库、C3P0数据库连接池,Java反射等

    MySQL数据库,JDBC接口,MyBatis框架等,掌握的数据的存放和管理. Java数据库开发基础,介绍MySQL数据库.JDBC操作数据库.C3P0数据库连接池,Java反射等内容,进行油画商城 ...

  5. MS SQL Server 数据库连接字符串详解

    MS SQL Server 数据库连接字符串详解 问题 : 超时时间已到.在从池中获取连接之前超时时间已过.出现这种情况可能是因为所有池连接都已被使用并已达到最大池大小. 解决办法 1. 在代码里面 ...

  6. Spring Boot 使用 Druid 连接池详解

    Spring Boot 使用 Druid 连接池详解 Alibaba Druid 是一个 JDBC 组件库,包含数据库连接池.SQL Parser 等组件,被大量业务和技术产品使用或集成,经历过严苛的 ...

  7. Java线程池详解学习:ThreadPoolExecutor

    Java线程池详解学习:ThreadPoolExecutor Java的源码下载参考这篇文章:Java源码下载和阅读(JDK1.8) - zhangpeterx的博客 在源码的目录java/util/ ...

  8. Java 线程池详解学习:FixedThreadPool,CachedThreadPool,ScheduledThreadPool...

    Java常用的线程池有FixedThreadPool和CachedThreadPool,我们可以通过查看他们的源码来进行学习. Java的源码下载参考这篇文章:Java源码下载和阅读(JDK1.8) ...

  9. c3p0 数据库连接池

    C3P0连接池 c3p0是一个开源的JDBC连接池,它实现了数据源和JNDI绑定,支持JDBC3规范和JDBC2的标准扩展.c3p0一般是与Hibernate,Spring等框架一块使用的,当然也可以 ...

  10. Java多线程之线程池详解

    Java多线程之线程池详解 目录: 线程池使用及优势 线程池3个常用方式 线程池7大参数深入介绍 线程池底层工作原理 1. 线程池使用及优势 线程池做的工作主要是控制运行的线程的数量,处理过程中将任务 ...

最新文章

  1. 微信浏览器下拉黑边的终极解决方案---wScroollFix
  2. python怎么拆分没有分隔符字符串_python如何拆分含有多种分隔符的字符串
  3. 前端如何获取联通积分_怎么利用定制的小程序来获取流量
  4. python画画-用python实现你的绘画梦想
  5. 数据库-优化-数据库结构的优化-表范式化优化
  6. POJ 1185 炮兵阵地
  7. 一个支持 CodeFirst/DbFirst/ModelFirst 的数据库小工具
  8. vc6开发一个抓包软件_开发一个软件多少钱?3种软件开发公司报价
  9. 学编程必备的三个网站
  10. Facebook妥协了,React回归
  11. 8tupian图片加密平台源码 v3.5
  12. javascript 应用_如何利用JavaScript的功能使您的应用脱机工作
  13. 解决ssh登录后闲置时间过长而断开连接
  14. 如何通过方法(函数)来实现两个基本数据类型的数值交换
  15. docker使用官方仓库上传与下拉images
  16. 新手如何Reverces(3自动化逆向篇)
  17. 刘宇凡:坚持就是写文章的最大技巧
  18. 手机android怎么开机画面,Android使用BroadcastReceiver实现手机开机之后显示画面的功能...
  19. ssh远程No route to host问题解决
  20. CentOS支持中文

热门文章

  1. c语言编程题及答案汇总,C语言编程题及参考答案解析.doc
  2. Python对微信好友进行简单统计分析
  3. c# 条码打印—电子监管码打印
  4. kanzi学习之路(序)
  5. R语言技巧:读取sas软件的sas格式文件
  6. Postgre SQL group_concat 写法
  7. Norton推出基于云查杀免费小工具Norton Power Eraser
  8. 阿里云OSS服务使用操作流程
  9. ijkplayer实现图形字幕的播放
  10. db107s-ASEMI整流桥堆怎么测量好坏,万用表测量DB107S四脚