java 對(duì)象池 commons-pool2 的使用(創(chuàng)建influxDB連接池)

參考引用文檔:http://www.reibang.com/p/b45b96a61451
https://blog.csdn.net/qq_37186947/article/details/104227552

書接上文 Java連接influxDB非迹,創(chuàng)建一個(gè)influxDB的連接池后控。

1瑟俭、添加依賴

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

2、實(shí)現(xiàn)PooledObjectFactory燕锥,創(chuàng)建一個(gè)對(duì)象工廠

需要重寫5個(gè)函數(shù):

  • makeObject 方法 :用于對(duì)象的新建顽频,一般是 new 出來之后包裝一下菱蔬。而什么時(shí)候需要新建呢,根據(jù)策略不同則時(shí)機(jī)不同砾层。例如在沒有閑置資源對(duì)象漩绵,且已存在的資源數(shù)不超過所設(shè)置的最大資源時(shí)新建。

  • destroyObject 方法:銷毀一個(gè)對(duì)象肛炮,除了很容易想到的閑置過長(zhǎng)時(shí)間被清理掉了導(dǎo)致需要銷毀之外止吐,還有如果進(jìn)行了第三個(gè)方法且返回了 false ,那么也是需要銷毀的侨糟。

  • validateObject方法:檢驗(yàn)這個(gè)對(duì)象是否還有有效碍扔,借出和歸還時(shí),以及內(nèi)置后臺(tái)線程檢測(cè)閑置情況時(shí)秕重,可以通過驗(yàn)證可以去除一些不符合業(yè)務(wù)邏輯的資源對(duì)象不同。默認(rèn)這個(gè)方法是不被調(diào)用的,要開啟則需要在PoolConfig中設(shè)置setTestOnBorrow , setTestOnReturn , setTestWhileIdle等屬性溶耘。

  • activeObject 方法: 在借用一個(gè)對(duì)象的時(shí)候調(diào)用二拐,則可以在此重置其內(nèi)部狀態(tài),那么返回的對(duì)象就像新的一樣

  • passivateObject 方法: 對(duì)應(yīng) activateObject 方法凳兵,是在歸還一個(gè)對(duì)象的時(shí)候調(diào)用卓鹿,注意不應(yīng)與activateObject方法有業(yè)務(wù)邏輯上的沖突

import com.influxdb.client.InfluxDBClient;
import com.influxdb.client.InfluxDBClientFactory;
import com.influxdb.client.domain.HealthCheck;
import org.apache.commons.pool2.PooledObject;
import org.apache.commons.pool2.PooledObjectFactory;
import org.apache.commons.pool2.impl.DefaultPooledObject;

/**
 * @author Jenson
 */
public class InfluxdbPooledObjectFactory implements PooledObjectFactory<InfluxDBClient> {

    /**
     * 重新初始化要由池返回的實(shí)例-即從池中借用一個(gè)對(duì)象時(shí)調(diào)用
     *
     * @param pooledObject 一個(gè)PooledObject包裝要激活的實(shí)例
     * @throws Exception
     */
    @Override
    public void activateObject(PooledObject<InfluxDBClient> pooledObject) throws Exception {
        System.out.println("重新初始化要由池返回的實(shí)例-即從池中借用一個(gè)對(duì)象時(shí)調(diào)用");
    }

    /**
     * 使用默認(rèn) (NORMAL) DestroyMode 銷毀池不再需要的實(shí)例。
     *
     * @param pooledObject
     * @throws Exception
     */
    @Override
    public void destroyObject(PooledObject<InfluxDBClient> pooledObject) throws Exception {
        InfluxDBClient influxDBClient = pooledObject.getObject();
        influxDBClient.close();
    }

    /**
     * 創(chuàng)建可由池提供服務(wù)的實(shí)例留荔,并將其包裝在由池管理的PooledObject中
     *
     * @return
     * @throws Exception
     */
    @Override
    public PooledObject<InfluxDBClient> makeObject() throws Exception {
        // 生成客戶端
//        InfluxDBClient client = InfluxDBClientFactory.create(this.url, this.token.toCharArray());
        InfluxDBClient client = InfluxDBClientFactory.create("http://127.0.0.1:8086",
                "S9dKiPFvnSUd9p406aCkCdwZ_q5rke8mNWzxERaNOL6SW0Fl2JIHZ1V9B317BqmBxzdaFwxei2TcYwJu2RXnOg==".toCharArray());
        System.out.println("創(chuàng)建可由池提供服務(wù)的實(shí)例,并將其包裝在由池管理的PooledObject中, hashcode :"+client.hashCode());
        return new DefaultPooledObject<>(client);
    }

    /**
     * 取消初始化要返回到空閑對(duì)象池的實(shí)例-即從池中歸還一個(gè)對(duì)象時(shí)調(diào)用
     *
     * @param pooledObject
     * @throws Exception
     */
    @Override
    public void passivateObject(PooledObject<InfluxDBClient> pooledObject) throws Exception {
        System.out.println("取消初始化要返回到空閑對(duì)象池的實(shí)例-即從池中歸還一個(gè)對(duì)象時(shí)調(diào)用");
    }

    /**
     * 確保實(shí)例可以安全地由池返回澜倦。
     *
     * @param pooledObject
     * @return 如果obj無效并且應(yīng)該從池中刪除聚蝶,則為false ,否則為true
     */
    @Override
    public boolean validateObject(PooledObject<InfluxDBClient> pooledObject) {
        InfluxDBClient influxDBClient = pooledObject.getObject();
        HealthCheck health = influxDBClient.health();
        return HealthCheck.StatusEnum.PASS.equals(health.getStatus());
    }
}

3藻治、創(chuàng)建對(duì)象池碘勉,繼承GenericObjectPool

import com.influxdb.client.InfluxDBClient;
import org.apache.commons.pool2.PooledObjectFactory;
import org.apache.commons.pool2.impl.AbandonedConfig;
import org.apache.commons.pool2.impl.GenericObjectPool;
import org.apache.commons.pool2.impl.GenericObjectPoolConfig;

/**
 * @author Jenson
 */
public class InluxdbClientPool extends GenericObjectPool<InfluxDBClient> {
    public InluxdbClientPool(PooledObjectFactory<InfluxDBClient> factory) {
        super(factory);
    }

    public InluxdbClientPool(PooledObjectFactory<InfluxDBClient> factory, GenericObjectPoolConfig<InfluxDBClient> config) {
        super(factory, config);
    }

    public InluxdbClientPool(PooledObjectFactory<InfluxDBClient> factory, GenericObjectPoolConfig<InfluxDBClient> config, AbandonedConfig abandonedConfig) {
        super(factory, config, abandonedConfig);
    }
}

4、創(chuàng)建對(duì)象池自動(dòng)裝配配置類桩卵,將對(duì)象池做成一個(gè)Bean

import com.influxdb.client.InfluxDBClient;
import com.jenson.factory.InfluxdbPooledObjectFactory;
import com.jenson.pool.InluxdbClientPool;
import org.apache.commons.pool2.impl.GenericObjectPoolConfig;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.annotation.PreDestroy;

/**
 * 對(duì)象池自動(dòng)裝配
 *
 * @author Jenson
 */
@Configuration
public class InfluxdbPoolAutoConfig {

    private InluxdbClientPool pool;

    @ConditionalOnClass({InfluxdbPooledObjectFactory.class})
    @Bean("inluxdbClientPool")
    protected InluxdbClientPool createInluxdbClientPool(){
        InfluxdbPooledObjectFactory factory = new InfluxdbPooledObjectFactory();
        // 設(shè)置對(duì)象池相關(guān)參數(shù)
        GenericObjectPoolConfig<InfluxDBClient> poolConfig = new GenericObjectPoolConfig<>();
        /**
         * 最大空閑
         */
        poolConfig.setMaxIdle(5);
        /**
         * 最大總數(shù)
         */
        poolConfig.setMaxTotal(10);
        /**
         * 最小空閑
         */
        poolConfig.setMinIdle(2);
        poolConfig.setBlockWhenExhausted(true);
        poolConfig.setTestOnBorrow(true);
        poolConfig.setTestOnReturn(true);
        poolConfig.setTestWhileIdle(true);
        poolConfig.setTimeBetweenEvictionRunsMillis(1000 * 60 * 30);
        //一定要關(guān)閉jmx验靡,不然springboot啟動(dòng)會(huì)報(bào)已經(jīng)注冊(cè)了某個(gè)jmx的錯(cuò)誤
        poolConfig.setJmxEnabled(false);

        // 新建一個(gè)對(duì)象池,傳入對(duì)象工廠和配置
        pool = new InluxdbClientPool(factory, poolConfig);

        initPool(3, 5);

        return pool;
    }

    /**
     * 預(yù)先加載testObject對(duì)象到對(duì)象池中
     *
     * @param initialSize 初始化連接數(shù)
     * @param maxIdle     最大空閑連接數(shù)
     */
    private void initPool(int initialSize, int maxIdle) {
        if (initialSize <= 0) {
            return;
        }

        int size = Math.min(initialSize, maxIdle);
        for (int i = 0; i < size; i++) {
            try {
                pool.addObject();
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }
    }

    @PreDestroy
    public void destroy() {
        if (pool != null) {
            pool.close();
        }
    }
}


5、使用池

從池中取出對(duì)象使用雏节,結(jié)束后再將對(duì)象歸還

@GetMapping("/get-influx-data")
    public List<FluxRecord> getInfluxData(@RequestParam Integer startTime) {
        String query = "from(bucket: \"zzs\") |> range(start: -"+startTime+"h) |> filter(fn: (r) => r._measurement == \"mem\")";
        List<FluxTable> tables = new ArrayList<>();
//        tables = influxdbClient.getQueryApi().query(query, "jenson");

        InfluxDBClient poolClient = null;
        try {
            poolClient = inluxdbClientPool.borrowObject();
            System.out.println("poolClient hashcode : "+poolClient.hashCode());
            tables = poolClient.getQueryApi().query(query, "jenson");
        } catch (Exception e) {
            e.printStackTrace();
        }
        finally {
            if (poolClient != null) {
                //最終歸還對(duì)象到對(duì)象池
                inluxdbClientPool.returnObject(poolClient);
            }
        }

        List<FluxRecord> fluxRecords = new ArrayList<>();
        tables.forEach(fluxTable -> {
            List<FluxRecord> records = fluxTable.getRecords();
            fluxRecords.addAll(records);
        });
        return fluxRecords;
    }

如果注釋掉歸還對(duì)象的這一步胜嗓,可以看到每次調(diào)用接口控制它輸出對(duì)象的hashcode都是不同的,超過三次后就會(huì)新建對(duì)象钩乍,當(dāng)達(dá)到最大MaxTotal時(shí)辞州,程序會(huì)等待直到有對(duì)象歸還到池中,所以從池中取出對(duì)象使用完后務(wù)必要將對(duì)象歸還寥粹。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末变过,一起剝皮案震驚了整個(gè)濱河市埃元,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌媚狰,老刑警劉巖岛杀,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異崭孤,居然都是意外死亡类嗤,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門裳瘪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來土浸,“玉大人,你說我怎么就攤上這事彭羹』埔粒” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵派殷,是天一觀的道長(zhǎng)还最。 經(jīng)常有香客問我,道長(zhǎng)毡惜,這世上最難降的妖魔是什么拓轻? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮经伙,結(jié)果婚禮上扶叉,老公的妹妹穿的比我還像新娘。我一直安慰自己帕膜,他們只是感情好枣氧,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著垮刹,像睡著了一般达吞。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上荒典,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天酪劫,我揣著相機(jī)與錄音,去河邊找鬼寺董。 笑死覆糟,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的遮咖。 我是一名探鬼主播搪桂,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了踢械?” 一聲冷哼從身側(cè)響起酗电,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎内列,沒想到半個(gè)月后撵术,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡话瞧,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年嫩与,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片交排。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡划滋,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出埃篓,到底是詐尸還是另有隱情处坪,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布架专,位于F島的核電站同窘,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏部脚。R本人自食惡果不足惜想邦,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望委刘。 院中可真熱鬧丧没,春花似錦、人聲如沸锡移。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽罩抗。三九已至,卻和暖如春灿椅,著一層夾襖步出監(jiān)牢的瞬間套蒂,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來泰國(guó)打工茫蛹, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留操刀,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓婴洼,卻偏偏與公主長(zhǎng)得像骨坑,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容