文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java中对象池怎么实现

2023-07-02 16:48

关注

这篇“Java中对象池怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java中对象池怎么实现”文章吧。

1. 什么是对象池

池化并不是什么新鲜的技术,它更像一种软件设计模式,主要功能是缓存一组已经初始化的对象,以供随时可以使用。对象池大多数场景下都是缓存着创建成本过高或者需要重复创建使用的对象,从池子中取对象的时间是可以预测的,但是新建一个对象的时间是不确定的。

当需要一个新对象时,就向池中借出一个,然后对象池标记当前对象正在使用,使用完毕后归还到对象池,以便再次借出。

常见的使用对象池化场景:

2. 为什么需要对象池

如果一个对象的创建成本很高,比如建立数据库的连接时耗时过长,在不使用池化技术的情况下,我们的查询过程可能是这样的。

在这种模式下,每次查询都要重新建立关闭连接,因为建立连接是一个耗时的操作,所以这种模式会影响程序的总体性能。

那么使用池化思想是怎么样的呢?同样的过程会转变成下面的步骤。

使用池化思想后,数据库连接并不会频繁的创建关闭,而是启动后就初始化了 N 个连接以供后续使用,使用完毕后归还对象,这样程序的总体性能得到提升。

3. 对象池的实现

通过上面的例子也可以发现池化思想的几个关键步骤:初始化、借出、归还。上面没有展示销毁步骤, 某些场景下还需要对象的销毁这一过程,比如释放连接。

下面我们手动实现一个简陋的对象池,加深下对对象池的理解。主要是定一个对象池管理类,然后在里面实现对象的初始化、借出、归还、销毁等操作。

package com.wdbyet.tool.objectpool.mypool;import java.io.Closeable;import java.io.IOException;import java.util.HashSet;import java.util.Stack;public class MyObjectPool<T extends Closeable> {    // 池子大小    private Integer size = 5;    // 对象池栈。后进先出    private Stack<T> stackPool = new Stack<>();    // 借出的对象的 hashCode 集合    private HashSet<Integer> borrowHashCodeSet = new HashSet<>();        public synchronized void addObj(T t) {        if ((stackPool.size() + borrowHashCodeSet.size()) == size) {            throw new RuntimeException("池中对象已经达到最大值");        }        stackPool.add(t);        System.out.println("添加了对象:" + t.hashCode());    }        public synchronized T borrowObj() {        if (stackPool.isEmpty()) {            System.out.println("没有可以被借出的对象");            return null;        }        T pop = stackPool.pop();        borrowHashCodeSet.add(pop.hashCode());        System.out.println("借出了对象:" + pop.hashCode());        return pop;    }        public synchronized void returnObj(T t) {        if (borrowHashCodeSet.contains(t.hashCode())) {            stackPool.add(t);            borrowHashCodeSet.remove(t.hashCode());            System.out.println("归还了对象:" + t.hashCode());            return;        }        throw new RuntimeException("只能归还从池中借出的对象");    }        public synchronized void destory() {        if (!borrowHashCodeSet.isEmpty()) {            throw new RuntimeException("尚有未归还的对象,不能关闭所有对象");        }        while (!stackPool.isEmpty()) {            T pop = stackPool.pop();            try {                pop.close();            } catch (IOException e) {                throw new RuntimeException(e);            }        }        System.out.println("已经销毁了所有对象");    }}

代码还是比较简单的,只是简单的示例,下面我们通过池化一个 Redis 连接对象 Jedis 来演示如何使用。

其实 Jedis 中已经有对应的 Jedis 池化管理对象了 JedisPool 了,不过我们这里为了演示对象池的实现,就不使用官方提供的 JedisPool 了。

启动一个 Redis 服务这里不做介绍,假设你已经有了一个 Redis 服务,下面引入 Java 中连接 Redis 需要用到的 Maven 依赖。

<dependency>    <groupId>redis.clients</groupId>    <artifactId>jedis</artifactId>    <version>4.2.0</version></dependency>

正常情况下 Jedis 对象的使用方式:

Jedis jedis = new Jedis("localhost", 6379);String name = jedis.get("name");System.out.println(name);jedis.close();

如果使用上面的对象池,就可以像下面这样使用。

package com.wdbyet.tool.objectpool.mypool;import redis.clients.jedis.Jedis;public class MyObjectPoolTest {    public static void main(String[] args) {        MyObjectPool<Jedis> objectPool = new MyObjectPool<>();        // 增加一个 jedis 连接对象        objectPool.addObj(new Jedis("127.0.0.1", 6379));        objectPool.addObj(new Jedis("127.0.0.1", 6379));        // 从对象池中借出一个 jedis 对象        Jedis jedis = objectPool.borrowObj();        // 一次 redis 查询        String name = jedis.get("name");        System.out.println(String.format("redis get:" + name));        // 归还 redis 连接对象        objectPool.returnObj(jedis);        // 销毁对象池中的所有对象        objectPool.destory();        // 再次借用对象        objectPool.borrowObj();    }}

输出日志:

添加了对象:1556956098
添加了对象:1252585652
借出了对象:1252585652
redis get:www.wdbyte.com
归还了对象:1252585652
已经销毁了所有对象
没有可以被借出的对象

如果使用 JMH 对使用对象池化进行 Redis 查询,和正常创建 Redis 连接然后查询关闭连接的方式进行性能对比,会发现两者的性能差异很大。下面是测试结果,可以发现使用对象池化后的性能是非池化方式的 5 倍左右。

Benchmark                   Mode  Cnt      Score       Error  Units
MyObjectPoolTest.test      thrpt   15   2612.689 &plusmn;   358.767  ops/s
MyObjectPoolTest.testPool  thrpt    9  12414.228 &plusmn; 11669.484  ops/s

4. 开源的对象池工具

上面自己实现的对象池总归有些简陋了,其实开源工具中已经有了非常好用的对象池的实现,如 Apache 的 commons-pool2 工具,很多开源工具中的对象池都是基于此工具实现,下面介绍这个工具的使用方式。

maven 依赖:

<dependency>    <groupId>org.apache.commons</groupId>    <artifactId>commons-pool2</artifactId>    <version>2.11.1</version></dependency>

在 commons-pool2 对象池工具中有几个关键的类。

下面通过一个具体的示例演示 commons-pool2 工具类的使用,这里依旧选择 Redis 连接对象 Jedis 作为演示。

实现 PooledObjectFactory 工厂类,实现其中的对象创建和销毁方法。

public class MyPooledObjectFactory implements PooledObjectFactory<Jedis> {    @Override    public void activateObject(PooledObject<Jedis> pooledObject) throws Exception {    }    @Override    public void destroyObject(PooledObject<Jedis> pooledObject) throws Exception {        Jedis jedis = pooledObject.getObject();        jedis.close();          System.out.println("释放连接");    }    @Override    public PooledObject<Jedis> makeObject() throws Exception {        return new DefaultPooledObject(new Jedis("localhost", 6379));    }    @Override    public void passivateObject(PooledObject<Jedis> pooledObject) throws Exception {    }    @Override    public boolean validateObject(PooledObject<Jedis> pooledObject) {        return false;    }}

继承 GenericObjectPool 类,实现对对象的借出、归还等操作。

public class MyGenericObjectPool extends GenericObjectPool<Jedis> {    public MyGenericObjectPool(PooledObjectFactory factory) {        super(factory);    }    public MyGenericObjectPool(PooledObjectFactory factory, GenericObjectPoolConfig config) {        super(factory, config);    }    public MyGenericObjectPool(PooledObjectFactory factory, GenericObjectPoolConfig config,        AbandonedConfig abandonedConfig) {        super(factory, config, abandonedConfig);    }}

可以看到 MyGenericObjectPool 类的构造函数中的入参有 GenericObjectPoolConfig 对象,这是个对象池的配置对象,可以配置对象池的容量大小等信息,这里就不配置了,使用默认配置。

通过 GenericObjectPoolConfig 的源码可以看到默认配置中,对象池的容量是 8 个

public class GenericObjectPoolConfig<T> extends BaseObjectPoolConfig<T> {        public static final int DEFAULT_MAX_TOTAL = 8;        public static final int DEFAULT_MAX_IDLE = 8;

下面编写一个对象池使用测试类。

public class ApachePool {    public static void main(String[] args) throws Exception {        MyGenericObjectPool objectMyObjectPool = new MyGenericObjectPool(new MyPooledObjectFactory());        Jedis jedis = objectMyObjectPool.borrowObject();        String name = jedis.get("name");        System.out.println(name);        objectMyObjectPool.returnObject(jedis);        objectMyObjectPool.close();    }}

输出日志:

redis get:www.wdbyte.com
释放连接

上面已经演示了 commons-pool2 工具中的对象池的使用方式,从上面的例子中可以发现这种对象池中只能存放同一种初始化条件的对象,如果这里的 Redis 我们需要存储一个本地连接和一个远程连接的两种 Jedis 对象,就不能满足了。那么怎么办呢?

其实 commons-pool2 工具已经考虑到了这种情况,通过增加一个 key 值可以在同一个对象池管理中进行区分,代码和上面类似,直接贴出完整的代码实现。

package com.wdbyet.tool.objectpool.apachekeyedpool;import org.apache.commons.pool2.BaseKeyedPooledObjectFactory;import org.apache.commons.pool2.KeyedPooledObjectFactory;import org.apache.commons.pool2.PooledObject;import org.apache.commons.pool2.impl.AbandonedConfig;import org.apache.commons.pool2.impl.DefaultPooledObject;import org.apache.commons.pool2.impl.GenericKeyedObjectPool;import org.apache.commons.pool2.impl.GenericKeyedObjectPoolConfig;import redis.clients.jedis.Jedis;public class ApacheKeyedPool {    public static void main(String[] args) throws Exception {        String key = "local";        MyGenericKeyedObjectPool objectMyObjectPool = new MyGenericKeyedObjectPool(new MyKeyedPooledObjectFactory());        Jedis jedis = objectMyObjectPool.borrowObject(key);        String name = jedis.get("name");        System.out.println("redis get :" + name);        objectMyObjectPool.returnObject(key, jedis);    }}class MyKeyedPooledObjectFactory extends BaseKeyedPooledObjectFactory<String, Jedis> {    @Override    public Jedis create(String key) throws Exception {        if ("local".equals(key)) {            return new Jedis("localhost", 6379);        }        if ("remote".equals(key)) {            return new Jedis("192.168.0.105", 6379);        }        return null;    }    @Override    public PooledObject<Jedis> wrap(Jedis value) {        return new DefaultPooledObject<>(value);    }}class MyGenericKeyedObjectPool extends GenericKeyedObjectPool<String, Jedis> {    public MyGenericKeyedObjectPool(KeyedPooledObjectFactory<String, Jedis> factory) {        super(factory);    }    public MyGenericKeyedObjectPool(KeyedPooledObjectFactory<String, Jedis> factory,        GenericKeyedObjectPoolConfig<Jedis> config) {        super(factory, config);    }    public MyGenericKeyedObjectPool(KeyedPooledObjectFactory<String, Jedis> factory,        GenericKeyedObjectPoolConfig<Jedis> config, AbandonedConfig abandonedConfig) {        super(factory, config, abandonedConfig);    }}

输出日志:

redis get :www.wdbyte.com

5. JedisPool 对象池实现分析

这篇文章中的演示都使用了 Jedis 连接对象,其实在 Jedis SDK 中已经实现了相应的对象池,也就是我们常用的 JedisPool 类。那么这里的 JedisPool 是怎么实现的呢?我们先看一下 JedisPool 的使用方式。

package com.wdbyet.tool.objectpool;import redis.clients.jedis.Jedis;import redis.clients.jedis.JedisPool;public class JedisPoolTest {    public static void main(String[] args) {        JedisPool jedisPool = new JedisPool("localhost", 6379);        // 从对象池中借一个对象        Jedis jedis = jedisPool.getResource();        String name = jedis.get("name");        System.out.println("redis get :" + name);        jedis.close();        // 彻底退出前,关闭 Redis 连接池        jedisPool.close();    }}

代码中添加了注释,可以看到通过 jedisPool.getResource() 拿到了一个对象,这里和上面 commons-pool2 工具中的 borrowObject 十分相似,继续追踪它的代码实现可以看到下面的代码。

// redis.clients.jedis.JedisPool// public class JedisPool extends Pool<Jedis> {public Jedis getResource() {    Jedis jedis = (Jedis)super.getResource();    jedis.setDataSource(this);    return jedis;}// 继续追踪 super.getResource()// redis.clients.jedis.util.Poolpublic T getResource() {    try {        return super.borrowObject();    } catch (JedisException var2) {        throw var2;    } catch (Exception var3) {        throw new JedisException("Could not get a resource from the pool", var3);    }}

竟然看到了 super.borrowObject() ,多么熟悉的方法,继续分析代码可以发现 Jedis 对象池也是使用了 commons-pool2 工具作为实现。既然如此,那么 jedis.close() 方法的逻辑我们应该也可以猜到了,应该有一个归还的操作,查看代码发现果然如此。

// redis.clients.jedis.JedisPool// public class JedisPool extends Pool<Jedis> {public void close() {    if (this.dataSource != null) {        Pool<Jedis> pool = this.dataSource;        this.dataSource = null;        if (this.isBroken()) {            pool.returnBrokenResource(this);        } else {            pool.returnResource(this);        }    } else {        this.connection.close();    }}// 继续追踪 super.getResource()// redis.clients.jedis.util.Poolpublic void returnResource(T resource) {    if (resource != null) {        try {            super.returnObject(resource);        } catch (RuntimeException var3) {            throw new JedisException("Could not return the resource to the pool", var3);        }    }}

通过上面的分析,可见 Jedis 确实使用了 commons-pool2 工具进行对象池的管理,通过分析 JedisPool 类的继承关系图也可以发现。

Java中对象池怎么实现

以上就是关于“Java中对象池怎么实现”这篇文章的内容,相信大家都有了一定的了解,希望小编分享的内容对大家有帮助,若想了解更多相关的知识内容,请关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯