深入理解Apache Commons Pool2池化技术

在这里插入图片描述

码到三十五 : 个人主页

心中有诗画,指尖舞代码,目光览世界,步履越千山,人间尽值得 !


在现代软件开发中,为了提高性能和资源利用率,开发者们经常使用池化技术来管理那些创建和销毁代价较高的对象,比如数据库连接、网络套接字或线程。Apache Commons Pool2是Apache基金会提供的一个优秀的对象池化库,它为开发者提供了一套丰富的API和灵活的配置选项,以实现对象的池化管理。

目录

    • 1️⃣Apache Commons Pool2简介
    • 2️⃣为什么要使用对象池
    • 3️⃣Apache Commons Pool2的工作原理
      • 3.1. 对象池(ObjectPool)
      • 3.2. 池化对象(PooledObject)
      • 3.3. 对象工厂(PooledObjectFactory)
    • 4️⃣对象的取用和回收
      • 4.1 对象的取用(Borrowing)
      • 4.2 对象的回收(Returning)
    • 5️⃣pache Commons Pool2实现数据库连接池
    • 6️⃣ Apache Commons Pool2的使用场景
      • 6.1. 数据库连接池
      • 6.2. HTTP连接池
      • 6.3. 线程池
      • 6.4. 其他需要复用对象的场景
    • 7️⃣结语

1️⃣Apache Commons Pool2简介

Apache Commons Pool2是Apache Commons下的一个开源项目,主要用于实现和管理对象池。对象池是一种常见的设计模式,通过复用来分摊昂贵对象的创建和销毁代价,从而优化资源利用和提高应用程序性能。

Commons Pool2提供了一套用于实现对象池化的API,并内置了多种各具特色的对象池实现。其被广泛应用在各种数据库连接池、线程池以及请求分发池中。其实现提供了一些参数来控制对象池的行为,例如最大池化对象数、最大空闲时间、最小空闲数等,可以根据不同的应用场景进行灵活配置。

此外,Commons Pool2也提供了一些常用的实现类,如GenericObjectPool,它实现了一个功能强大的对象池,可以方便地进行配置和扩展。通过使用Commons Pool2,开发者可以更加轻松地实现和管理对象池,提高应用程序的性能和可靠性。

2️⃣为什么要使用对象池

  1. 资源复用:对象池通过复用对象实例,避免了频繁创建和销毁对象带来的开销。这对于创建和销毁成本较高的对象(如数据库连接、线程、复杂的数据结构等)尤为有益。

  2. 性能提升:由于减少了对象的创建和销毁次数,应用程序的响应时间得以改善,整体性能得到提升。对象池可以确保在需要时快速提供可用对象,减少了等待时间。

  3. 降低垃圾收集压力:频繁的对象创建和销毁会增加垃圾收集器的工作负担,可能导致应用程序的停顿和延迟。对象池通过减少不必要的对象分配和释放,降低了垃圾收集的频率和强度,从而提高了应用程序的稳定性。

  4. 可预测性和可控性:对象池允许开发者对池中的对象数量进行控制和调整,以满足应用程序的需求。通过配置池的大小、最大空闲时间等参数,可以实现对资源使用的精细控制,提高系统的可预测性和可控性。

  5. 简化资源管理:对象池封装了对象的创建、验证、销毁等复杂逻辑,使得开发者可以更加专注于业务逻辑的实现,而无需过多关注底层的资源管理细节。

总之,对象池是一种有效的资源管理技术,可以帮助开发者提高应用程序的性能、稳定性和可维护性。然而,需要注意的是,对象池并不适用于所有场景。在决定是否使用对象池时,需要综合考虑对象的创建和销毁成本、资源消耗情况、并发需求等因素。

3️⃣Apache Commons Pool2的工作原理

Commons Pool2提供了一套用于实现对象池化的API,并内置了多种各具特色的对象池实现。其中,核心的接口是ObjectPool,它定义了对象池应该实现的行为,包括对象的取用(borrow)、回收(return)和其他管理操作。同时,PooledObject是对池中对象的封装,包含对象的状态和一些其他信息。PooledObjectFactory是一个工厂类,负责具体对象的创建、初始化、状态销毁和验证等工作。
在这里插入图片描述

其工作原理主要基于以上三个核心概念:对象池(ObjectPool)、池化对象(PooledObject)和对象工厂(PooledObjectFactory)。

3.1. 对象池(ObjectPool)

  • 定义了对象池应该实现的行为,包括对象的取用(borrow)、回收(return)和其他管理操作。
  • 对象池负责存储和管理所有池化对象。它内部维护了一个队列,用于存储空闲对象,并在需要时提供对象,当对象不再使用时将其回收。
  • 对象池还提供了一系列的配置参数,比如最大池化对象数、最小空闲对象数、最大等待时间等,这些参数可以帮助开发者根据应用场景来细粒度地调整对象池的行为。

3.2. 池化对象(PooledObject)

  • 池化对象是对实际对象的包装。它除了持有实际对象的引用外,还包含了一些元数据,比如对象的状态(空闲、使用中、待销毁等)、创建时间、最后使用时间等。
  • 当一个实际对象被包装成池化对象并加入到对象池中时,它的生命周期就交由对象池来管理。只有当对象池决定销毁该对象时,实际对象的生命周期才会结束。

3.3. 对象工厂(PooledObjectFactory)

  • 对象工厂负责创建和销毁池化对象。它提供了create()、destroy()和validate()等方法。
  • 当对象池需要一个新的对象时,它会调用对象工厂的create()方法来创建一个新的对象,并将其包装成池化对象后加入到对象池中。
  • 当对象池中的一个对象不再需要使用时,对象池会调用对象工厂的destroy()方法来销毁该对象。但在销毁之前,对象池会先调用validate()方法来检查该对象是否仍然可用。如果validate()方法返回false,则对象池会立即销毁该对象;否则,它会将该对象标记为空闲状态并放回到对象池中等待下次使用。

4️⃣对象的取用和回收

Apache Commons Pool2 对象池提供了对象的创建、验证、取用(borrowing)、回收(returning)和销毁等功能。对象池的主要目的是复用对象,以减少对象创建和销毁的开销。
以下是 Apache Commons Pool2 中对象的取用和回收逻辑:

4.1 对象的取用(Borrowing)

  1. 请求对象:当客户端需要从对象池中获取一个对象时,它会调用 ObjectPool.borrowObject() 方法。

  2. 检查空闲对象:池首先会检查是否有可用的空闲对象。这通常是通过查看一个内部队列或集合来实现的,该队列或集合维护着当前未被使用的对象。

  3. 验证对象:如果找到了一个空闲对象,池通常会使用 PooledObjectFactory.validateObject() 方法来验证该对象是否仍然有效。如果对象无效,它将被销毁,并且池会尝试获取另一个对象。

  4. 创建新对象(如果需要):如果没有可用的空闲对象,或者所有空闲对象都已失效,池将使用 PooledObjectFactory.create() 方法来创建一个新对象。如果创建失败(例如,由于资源限制或配置问题),则可能会抛出异常。

  5. 返回对象给客户端:一旦验证或创建了一个有效对象,它就会被返回给客户端以供使用。此时,该对象被视为“被借出”的状态。

4.2 对象的回收(Returning)

  1. 归还对象:当客户端完成对象的使用后,它应该调用 ObjectPool.returnObject() 方法来将对象归还给池。这是确保对象能够被其他客户端复用的重要步骤。

  2. 验证对象:与取用过程类似,归还的对象也会通过 PooledObjectFactory.validateObject() 方法进行验证。如果验证失败,对象将被销毁而不是放回池中。

  3. 放回空闲队列:如果对象验证成功,它将被放回池的空闲队列中,等待下一个客户端的请求。

  4. 处理过剩对象:在某些情况下,当池中的空闲对象数量超过配置的最大空闲数时,池可能会选择销毁一些对象以减少资源占用。这通常是通过 PooledObjectFactory.destroyObject() 方法来实现的。

  5. 资源清理:除了验证和放回对象外,归还过程还可能包括一些额外的资源清理步骤,如关闭数据库连接、释放网络资源等。这些步骤通常是在 PooledObjectFactory 的实现中定义的。

通过管理对象的生命周期和复用,Apache Commons Pool2 能够帮助应用程序提高性能并减少资源消耗。然而,正确配置和使用对象池是至关重要的,以避免出现资源泄漏、性能瓶颈或其他问题。

5️⃣pache Commons Pool2实现数据库连接池

下面代码使用Apache Commons Pool2实现一个简单的数据库连接池。这个示例将展示如何创建一个自定义的PooledObjectFactory来管理数据库连接,并配置和使用ObjectPool来复用这些连接。

首先,我们需要一个PooledObjectFactory实现,用于创建、验证和销毁数据库连接:

import org.apache.commons.pool2.BasePooledObjectFactory;
import org.apache.commons.pool2.PooledObject;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;public class DatabaseConnectionFactory extends BasePooledObjectFactory<Connection> {private String connectionString;private String username;private String password;public DatabaseConnectionFactory(String connectionString, String username, String password) {this.connectionString = connectionString;this.username = username;this.password = password;}// 创建新的数据库连接@Overridepublic Connection create() {try {return DriverManager.getConnection(connectionString, username, password);} catch (SQLException e) {throw new RuntimeException("无法创建数据库连接", e);}}// 销毁数据库连接@Overridepublic void destroyObject(PooledObject<Connection> p) throws Exception {p.getObject().close();}// 验证数据库连接是否有效@Overridepublic boolean validateObject(PooledObject<Connection> p) {try {return p.getObject().isValid(1); // 设置一个非常短的超时,仅用于检查连接是否仍然可用} catch (SQLException e) {return false;}}// 激活对象(可选实现,这里我们什么也不做)@Overridepublic void activateObject(PooledObject<Connection> p) throws Exception {// 可以在这里进行一些连接重新激活的操作,例如设置自动提交、隔离级别等}// 钝化对象(可选实现,这里我们什么也不做)@Overridepublic void passivateObject(PooledObject<Connection> p) throws Exception {// 可以在对象返回到池之前执行一些清理或重置操作}
}

接下来,我们需要配置和创建ObjectPool

import org.apache.commons.pool2.impl.GenericObjectPool;
import org.apache.commons.pool2.impl.GenericObjectPoolConfig;public class DatabaseConnectionPool {private static GenericObjectPool<Connection> pool;static {// 配置连接池的参数GenericObjectPoolConfig config = new GenericObjectPoolConfig();config.setMaxTotal(10); // 设置连接池的最大连接数config.setMaxIdle(5); // 设置连接池的最大空闲连接数config.setMinIdle(2); // 设置连接池的最小空闲连接数// 创建连接工厂DatabaseConnectionFactory factory = new DatabaseConnectionFactory("jdbc:mysql://localhost:3306/mydatabase", "user", "password");// 初始化连接池pool = new GenericObjectPool<>(factory, config);}// 获取数据库连接public static Connection getConnection() throws Exception {return pool.borrowObject();}// 归还数据库连接到池public static void releaseConnection(Connection conn) {if (conn != null) {pool.returnObject(conn);}}// 关闭连接池(通常在应用程序关闭时调用)public static void close() {if (pool != null) {pool.close();}}
}

最后,我们可以在应用程序中使用这个连接池来获取和释放数据库连接:

public class Application {public static void main(String[] args) {// 从连接池中获取连接try (Connection conn = DatabaseConnectionPool.getConnection()) {// 使用连接执行数据库操作// ...// 连接会在try-with-resources块结束时自动归还到池中} catch (Exception e) {// 处理异常e.printStackTrace();}// 注意:在应用程序结束时,应该调用DatabaseConnectionPool.close()来关闭连接池。}
}

在上面的示例中,DatabaseConnectionFactory类负责创建、验证和销毁数据库连接,而DatabaseConnectionPool类则负责配置和管理连接池。应用程序通过调用DatabaseConnectionPool.getConnection()来获取连接,并在使用完毕后通过DatabaseConnectionPool.releaseConnection(conn)来归还连接到池中。使用try-with-resources语句可以确保连接在使用完毕后被正确关闭并归还到池中,即使在执行数据库操作时发生异常也是如此。

这个示例演示了Apache Commons Pool2在实际应用程序中的一个典型用法,即通过对象池化管理来复用昂贵的资源,从而提高应用程序的性能和效率。

6️⃣ Apache Commons Pool2的使用场景

Apache Commons Pool2由于其高效的对象管理能力和灵活的配置选项,在多种场景中得到了广泛应用:

6.1. 数据库连接池

  • 在Web应用程序或后台服务中,经常需要频繁地与数据库进行交互。如果每次交互都创建一个新的数据库连接并在使用后立即销毁它,那么这将造成大量的资源浪费和时间开销。通过使用Apache Commons Pool2来实现数据库连接池,可以复用数据库连接对象,显著提高应用程序的性能和吞吐量。

6.2. HTTP连接池

  • 在处理大量HTTP请求时,为每个请求创建一个新的HTTP连接也是不划算的。通过使用Apache Commons Pool2来管理HTTP连接对象,可以避免频繁地建立和关闭连接所带来的开销,提高系统的并发处理能力。

6.3. 线程池

  • 线程是操作系统中的昂贵资源之一。频繁地创建和销毁线程会导致系统性能下降甚至崩溃。通过使用Apache Commons Pool2来实现线程池,可以复用已经创建的线程对象来处理任务队列中的任务,从而降低线程创建和销毁的开销,提高系统的稳定性和响应速度。

6.4. 其他需要复用对象的场景

  • 除了上述常见的应用场景外,Apache Commons Pool2还可以应用于其他任何需要复用对象的场景中,比如文件句柄池、套接字连接池等。只要对象的创建和销毁代价较高且需要频繁使用,就可以考虑使用Apache Commons Pool2来实现对象池化管理。

7️⃣结语

总的来说,Commons Pool2是一个成熟、稳定且易于使用的对象池化框架,它能够帮助开发者提高应用程序的性能和可靠性,降低资源消耗和垃圾收集的压力。无论是数据库连接池、线程池还是其他类型的对象池,Commons Pool2都是一个值得考虑的选择。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/535067.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SpringController返回值和异常自动包装

今天遇到一个需求&#xff0c;在不改动原系统代码的情况下。将Controller的返回值和异常包装到一个统一的返回对象中去。 例如原系统的接口 public String myIp(ApiIgnore HttpServletRequest request);返回的只是一个IP字符串"0:0:0:0:0:0:0:1"&#xff0c;目前接口…

t_t 你们到底是从哪儿找的那种拿我当制杖一样教的简单的纯HTML模板的呀?

在这个万物vue的年代&#xff0c;网页设计越来越框架化。 上网搜个资料学习学习吧&#xff0c;咵咵咵&#xff0c;“游泳健身&#xff0c;vue了解一下” 我只是想简单地学个html&#xff0c;js啊&#xff01;怎么就这么复杂&#xff01; 曾几何时&#xff0c;在网上找个网页…

I2C学习总结

i2c概述 I2C&#xff08;Inter-Intergreted Circuit&#xff09; 是一种串行通信协议&#xff0c;用于集成电路之间完成数据传输&#xff0c;i2c用广泛用以各种领域&#xff0c;包括电子设备、嵌入式系统、工业自动化等&#xff1b; i2c仅仅只是一个数据传输的协议&#xff0c…

Splitpanes拆分窗格插件使用

目录 基本用法 纵向排列 遍历渲染 动态拆分宽度 项目开发中用到了拆分窗格(就是下面的效果&#xff0c;可以拆分网页&#xff0c;我们项目通常都是用左右两块拆分&#xff0c;可以通过拖动图标进行左右拖动)&#xff0c;于是就发现了一个很好用的插件&#xff1a;Splitpane…

2024年AI辅助研发:科技创新的引擎

CSND - 个人主页&#xff1a;17_Kevin-CSDN博客 收录专栏&#xff1a;《人工智能》 技术进展 进入2024年&#xff0c;人工智能&#xff08;AI&#xff09;在科技界和工业界的焦点地位更加巩固&#xff0c;其在辅助研发领域的技术进步尤为显著。深度学习技术的突飞猛进使得数据分…

1.Python是什么?——《跟老吕学Python编程》

1.Python是什么&#xff1f;——《跟老吕学Python编程》 Python是一种什么样的语言&#xff1f;Python的优点Python的缺点 Python发展历史Python的起源Python版本发展史 Python的价值学Python可以做什么职业&#xff1f;Python可以做什么应用&#xff1f; Python是一种什么样的…

图片格式转换怎么操作?这一个方法快快收藏

图片格式转换能够改变图片的质量、大小兼容性。不同的图片格式用途也不同&#xff0c;当我们需要转换图片格式的时候要怎么操作呢&#xff1f;下面&#xff0c;小编给大家分享一款操作简单&#xff0c;小白也能轻松上手的图片转换器&#xff08;https://www.yasuotu.com/geshi&…

“光谱视界革新:ChatGPT在成像光谱遥感中的智能革命“

遥感技术主要通过卫星和飞机从远处观察和测量我们的环境&#xff0c;是理解和监测地球物理、化学和生物系统的基石。ChatGPT是由OpenAI开发的最先进的语言模型&#xff0c;在理解和生成人类语言方面表现出了非凡的能力。本文重点介绍ChatGPT在遥感中的应用&#xff0c;人工智能…

【SpringBoot】整合Druid数据源和Mybatis 项目打包和运行

文章目录 一、整合Druid数据源二、整合Mybatis2.1 MyBatis整合步骤2.1 Mybatis整合实践2.1 声明式事务整合配置2.1 AOP整合配置 三、项目打包和运行命令启动和参数说明 总结web 与 springboot 打包区别JDK8的编译环境 执行17高版本jar 一、整合Druid数据源 创建模块 &#xff1…

BEVFormer代码运行笔记

1. 代码下载 git clone https://github.com/fundamentalvision/BEVFormer.git 2. 环境配置 使用conda创建环境 conda create -n open-mmlab python3.8 -y 进入环境 conda activate open-mmlab 允许使用pip安装依赖库 export PIP_REQUIRE_VIRTUALENVfalse 安装pytorch和…

FPGA - 单总线协议(one-wire)

1&#xff0c;简介 单总线&#xff08;one-wire&#xff09;是美国 DALLAS 公司推出的外围串行扩展总线技术&#xff0c;与 SPI、I2C 等串行数据通信方式不同&#xff0c;它采用单根信号线&#xff0c;既传输时钟又传输数据&#xff0c;而且数据传输是双向的。它具有节省 I/O口…

如何使用ChatGPT辅助写论文、数据分析、AI绘图?【附学习资料】

原文链接&#xff1a;如何使用ChatGPT辅助写论文、数据分析、AI绘图&#xff1f;【附学习资料】https://mp.weixin.qq.com/s?__bizMzUzNTczMDMxMg&mid2247597452&idx1&sn8aa991235ffee89fc76590a90b9005d7&chksmfa823c6bcdf5b57df0cb02ecd1821921f38ea6de34c7…