国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

如何快速實(shí)現(xiàn)一個(gè)連接池?

開(kāi)發(fā) 前端
在實(shí)際工作中,我們經(jīng)常會(huì)用到各種連接池,例如:連接 FTP 服務(wù)器的連接數(shù)有限,需要建立一個(gè)連接池;連接數(shù)據(jù)庫(kù)的連接數(shù)有限,需要建立一個(gè)連接池。那我們?nèi)绾稳タ焖賹?shí)現(xiàn)一個(gè)連接池呢?

[[401914]]

在實(shí)際工作中,我們經(jīng)常會(huì)用到各種連接池,例如:連接 FTP 服務(wù)器的連接數(shù)有限,需要建立一個(gè)連接池;連接數(shù)據(jù)庫(kù)的連接數(shù)有限,需要建立一個(gè)連接池。那我們?nèi)绾稳タ焖賹?shí)現(xiàn)一個(gè)連接池呢?

無(wú)論是 FTP 連接池,還是數(shù)據(jù)庫(kù)連接池,我們會(huì)發(fā)現(xiàn)它們都有相同的地方,它們都需要:生命周期管理、連接創(chuàng)建管理等等。如果我們從零開(kāi)始去實(shí)現(xiàn)這些功能,那我們要耗費(fèi)的時(shí)間就很長(zhǎng)了!那有沒(méi)有一個(gè)通用的庫(kù)可以快速實(shí)現(xiàn)一個(gè)線程池呢?

得益于 Java 完善的生態(tài),前人們針對(duì)這種需要開(kāi)發(fā)了一個(gè)通用庫(kù):Apache Commons Pool(下文簡(jiǎn)稱 ACP)。本質(zhì)上來(lái)說(shuō),ACP 庫(kù)提供的是管理對(duì)象池的通用能力,當(dāng)然也可以用來(lái)管理連接池了!

什么是 ACP?

ACP 庫(kù)提供了一整套用于實(shí)現(xiàn)對(duì)象池化的 API,以及若干種各具特色的對(duì)象池實(shí)現(xiàn)。目前最常用的版本是 2.0 版本,相對(duì)于 1.x 版本而言,并不是簡(jiǎn)單升級(jí)。2.0 版本是對(duì)象池實(shí)現(xiàn)的完全重寫(xiě),顯著的提升了性能和可伸縮性,并且包含可靠的實(shí)例跟蹤和池監(jiān)控。

Apache Commons Pool 的官網(wǎng)地址為:Pool – Overview,想翻找相關(guān)文檔資料,到這里去是最權(quán)威、最全面的。

如何使用 ACP?

要使用 ACP 實(shí)現(xiàn)一個(gè)線程池,首先需要先引入 ACP 的依賴包,這里以 Maven 為例。

  1. <dependency> 
  2.  <groupId>org.apache.commons</groupId> 
  3.  <artifactId>commons-pool2</artifactId> 
  4.  <version>2.0</version> 
  5. </dependency> 

 

要使用 ACP 實(shí)現(xiàn)一個(gè)對(duì)象池,大致可以分為三個(gè)步驟:

  • 創(chuàng)建對(duì)象工廠:告訴 ACP 如何創(chuàng)建你要的對(duì)象。
  • 創(chuàng)建對(duì)象池:告訴 ACP 你想創(chuàng)建一個(gè)怎樣的對(duì)象池。
  • 使用對(duì)象池:ACP 告訴你如何使用你的對(duì)象。

創(chuàng)建對(duì)象工廠

對(duì)象工廠告訴 ACP,它應(yīng)該如何去創(chuàng)建、激活、鈍化、銷毀你的對(duì)象。創(chuàng)建對(duì)象工廠非常簡(jiǎn)單,只需要實(shí)現(xiàn) ACP 的 PooledObjectFactory 接口即可。PooledObjectFactory 接口的定義如下:

  1. public interface PooledObjectFactory<T> { 
  2.   PooledObject<T> makeObject() throws Exception; 
  3.   void destroyObject(PooledObject<T> p) throws Exception; 
  4.   boolean validateObject(PooledObject<T> p); 
  5.   void activateObject(PooledObject<T> p) throws Exception; 
  6.   void passivateObject(PooledObject<T> p) throws Exception; 

但更多情況下,我們會(huì)繼承 BasePooledObjectFactory 類來(lái)實(shí)現(xiàn)對(duì)象工廠。因?yàn)?BasePooledObjectFactory 類是 PooledObjectFactory 的基礎(chǔ)實(shí)現(xiàn)類,使用它可以幫我們省了很多麻煩。通過(guò)繼承這個(gè)抽象類,我們只需要實(shí)現(xiàn)兩個(gè)方法:create() 和 wrap() 方法。

  1. // 告訴 ACP 如何創(chuàng)建對(duì)象 
  2. public abstract T create() throws Exception; 
  3. // 定義你要返回的對(duì)象 
  4. public abstract PooledObject<T> wrap(T obj); 

create() 方法定義你的對(duì)象初始化過(guò)程,最后將初始化完成的對(duì)象返回。例如你想定義一個(gè) SFTP 的連接,那么你首先需要定義一個(gè) JSch 對(duì)象,之后設(shè)置賬號(hào)密碼,之后連接服務(wù)器,最后返回一個(gè) ChannelSftp 對(duì)象。

  1. public ChannelSftp create() { 
  2.     // SFTP 連接的創(chuàng)建過(guò)程 

wrap() 方法定義你要返回的對(duì)象,對(duì)于一個(gè) SFTP 的連接池來(lái)說(shuō),其實(shí)就是一個(gè) ChannelSftp 對(duì)象。一般情況下可以使用類 DefaultPooledObject 替代,參考實(shí)現(xiàn)如下:

  1. @Override 
  2. public PooledObject<Foo> wrap(Foo foo) { 
  3.     return new DefaultPooledObject<Foo>(foo); 

創(chuàng)建對(duì)象池

創(chuàng)建好對(duì)象工廠之后,ACP 已經(jīng)知道你需要的對(duì)象如何創(chuàng)建了。那么接下來(lái),你需要根據(jù)你的實(shí)際需要,去創(chuàng)建一個(gè)對(duì)象池。在 ACP 中,我們通過(guò) GenericObjectPool 以及 GenericObjectPoolConfig 來(lái)創(chuàng)建一個(gè)對(duì)象池。

  1. // 聲明一個(gè)對(duì)象池 
  2. private GenericObjectPool<ChannelSftp> sftpConnectPool; 
  3.  
  4. // 設(shè)置連接池配置 
  5.         GenericObjectPoolConfig poolConfig = new GenericObjectPoolConfig(); 
  6.         poolConfig.setEvictionPolicyClassName("tech.shuyi.javacodechip.acp.SftpEvictionPolicy"); 
  7.         poolConfig.setBlockWhenExhausted(true); 
  8.         poolConfig.setJmxEnabled(false); 
  9.         poolConfig.setMaxWaitMillis(1000 * 10); 
  10.         poolConfig.setTimeBetweenEvictionRunsMillis(60 * 1000); 
  11.         poolConfig.setMinEvictableIdleTimeMillis(20 * 1000); 
  12.         poolConfig.setTestWhileIdle(true); 
  13.         poolConfig.setTestOnReturn(true); 
  14.         poolConfig.setTestOnBorrow(true); 
  15.         poolConfig.setMaxTotal(3); 
  16.         // 設(shè)置拋棄策略 
  17.         AbandonedConfig abandonedConfig = new AbandonedConfig(); 
  18.         abandonedConfig.setRemoveAbandonedOnMaintenance(true); 
  19.         abandonedConfig.setRemoveAbandonedOnBorrow(true); 
  20.         this.sftpConnectPool = new GenericObjectPool<>(sftpConnectFactory, poolConfig, abandonedConfig); 

在上面創(chuàng)建 SFTP 連接池的代碼中,我們配置了一些線程池的參數(shù)以及設(shè)置了拋棄策略。拋棄策略是非常重要的,如果沒(méi)有設(shè)置拋棄策略,那么會(huì)拿到失效的連接從而導(dǎo)致獲取文件失敗。拋棄策略是通過(guò) poolConfig.setEvictionPolicyClassName 來(lái)設(shè)置的,我們這里設(shè)置的是 SftpEvictionPolicy 類,其代碼內(nèi)容如下:

  1. @Slf4j 
  2. @Component 
  3. public class SftpEvictionPolicy implements EvictionPolicy<com.jcraft.jsch.ChannelSftp> { 
  4.     @Override 
  5.     public boolean evict(EvictionConfig config, PooledObject<com.jcraft.jsch.ChannelSftp> underTest, int idleCount) { 
  6.         try { 
  7.             // 連接失效時(shí)進(jìn)行驅(qū)逐 
  8.             if (!underTest.getObject().isConnected()) { 
  9.                 log.warn("connect time out, evict the connection. time={}",System.currentTimeMillis() - underTest.getLastReturnTime()); 
  10.                 return true
  11.             } 
  12.         }catch (Exception e){ 
  13.             return true
  14.         } 
  15.         return false
  16.     } 

看到這里,創(chuàng)建線程池的代碼就結(jié)束了,SftpConnectPool 文件的全部?jī)?nèi)容如下:

  1. @Slf4j 
  2. public class SftpConnectPool { 
  3.  
  4.     private GenericObjectPool<ChannelSftp> sftpConnectPool; 
  5.  
  6.     public SftpConnectPool(SftpConnectFactory sftpConnectFactory) { 
  7.         // 設(shè)置連接池配置 
  8.         GenericObjectPoolConfig poolConfig = new GenericObjectPoolConfig(); 
  9.         poolConfig.setEvictionPolicyClassName("tech.shuyi.javacodechip.acp.SftpEvictionPolicy"); 
  10.         poolConfig.setBlockWhenExhausted(true); 
  11.         poolConfig.setJmxEnabled(false); 
  12.         poolConfig.setMaxWaitMillis(1000 * 10); 
  13.         poolConfig.setTimeBetweenEvictionRunsMillis(60 * 1000); 
  14.         poolConfig.setMinEvictableIdleTimeMillis(20 * 1000); 
  15.         poolConfig.setTestWhileIdle(true); 
  16.         poolConfig.setTestOnReturn(true); 
  17.         poolConfig.setTestOnBorrow(true); 
  18.         poolConfig.setMaxTotal(3); 
  19.         // 設(shè)置拋棄策略 
  20.         AbandonedConfig abandonedConfig = new AbandonedConfig(); 
  21.         abandonedConfig.setRemoveAbandonedOnMaintenance(true); 
  22.         abandonedConfig.setRemoveAbandonedOnBorrow(true); 
  23.         this.sftpConnectPool = new GenericObjectPool<>(sftpConnectFactory, poolConfig, abandonedConfig); 
  24.     } 
  25.  
  26.     public ChannelSftp borrowObject() { 
  27.         try { 
  28.             return sftpConnectPool.borrowObject(); 
  29.         } catch (Exception e) { 
  30.             log.error("borrowObject error", e); 
  31.             return null
  32.         } 
  33.     } 
  34.  
  35.     public void returnObject(ChannelSftp channelSftp) { 
  36.         if (channelSftp!=null) { 
  37.             sftpConnectPool.returnObject(channelSftp); 
  38.         } 
  39.     } 

為了方便使用,我還增加了 borrowObject 和 returnObject 方法,但這兩個(gè)并不是必須的。在這兩個(gè)方法中,我們分別調(diào)用了 GenericObjectPool 類的 borrowObject 方法和 returnObject 方法。這正是 ACP 提供的、使用線程池對(duì)象的方法,先借一個(gè)對(duì)象,之后歸還對(duì)象。

注:其實(shí)在這一步,已經(jīng)包含了對(duì)象池的使用了。但實(shí)際使用的時(shí)候,我們經(jīng)常是將對(duì)象池的聲明與使用放在同一個(gè)類中,因此為了講解方便,這里沒(méi)有分開(kāi)。因此下文的使用對(duì)象池,本質(zhì)上是對(duì)對(duì)象池做進(jìn)一步封裝。

使用對(duì)象池

到這里我們的 SFTP 對(duì)象池就已經(jīng)創(chuàng)建完畢了,是不是非常簡(jiǎn)單呢!但在實(shí)際的工作中,我們通常會(huì)在這基礎(chǔ)上,做一些封裝。對(duì)于我們這次的 SFTP 連接池來(lái)說(shuō),我們會(huì)對(duì)外直接提供下載文件的服務(wù),將 SFTP 對(duì)象池進(jìn)一步封裝起來(lái),不需要關(guān)心怎么獲取文件。

  1. public class SftpFileHelper { 
  2.  
  3.     @Autowired 
  4.     private SftpConnectPool sftpConnectPool; 
  5.  
  6.     public void download(String dir, String file, String saveUrl)throws IOException { 
  7.         ChannelSftp sftp = sftpConnectPool.borrowObject(); 
  8.         log.info("begin to download file, dir={}, file={}, saveUrl={}", dir, file, saveUrl); 
  9.         try { 
  10.             if (!StringUtils.isEmpty(dir)) { 
  11.                 sftp.cd(dir); 
  12.             } 
  13.             File downloadFile = new File(saveUrl); 
  14.             sftp.get(file, new FileOutputStream(downloadFile)); 
  15.         }catch (Exception e){ 
  16.             log.warn("下載文件失敗", e); 
  17.         }finally { 
  18.             sftpConnectPool.returnObject(sftp); 
  19.         } 
  20.         log.info("file:{} is download successful", file); 
  21.     } 

最后我們寫(xiě)一個(gè)測(cè)試用例來(lái)試一試,是否能正常下載文件。

  1. @RunWith(SpringRunner.class) 
  2. @SpringBootTest 
  3. @Slf4j 
  4. public class SftpFileHelperTest { 
  5.  
  6.     @Autowired 
  7.     private SftpFileHelper sftpFileHelper; 
  8.  
  9.     @Test 
  10.     public void testDownloadFtpFile() throws Exception { 
  11.         sftpFileHelper.download("dir""fileName""fileName"); 
  12.     } 

沒(méi)有意外的話,你會(huì)看到一條綠線,文件已經(jīng)被成功下載了!

總結(jié)

本文針對(duì) Apache Commons Pool 庫(kù)最常用的對(duì)象池功能做了演示??赐赀@篇文章,我們知道創(chuàng)建一個(gè)線程池需要三個(gè)步驟,分別是:

創(chuàng)建對(duì)象工廠:告訴 ACP 如何創(chuàng)建你要的對(duì)象。

創(chuàng)建對(duì)象池:告訴 ACP 你想創(chuàng)建一個(gè)怎樣的對(duì)象池、設(shè)置驅(qū)逐策略。

使用對(duì)象池:ACP 告訴你如何使用你的對(duì)象。

本文相關(guān)代碼存放在博主 Github 項(xiàng)目:java-code-chip 中,可以點(diǎn)擊地址獲?。簀ava-code-chip/src/main/java/tech/shuyi/javacodechip/acp at master · chenyurong/java-code-chip

ACP 庫(kù)能夠讓讀者朋友們快速地創(chuàng)建一個(gè)對(duì)象池,更加專注于業(yè)務(wù)內(nèi)容。但事實(shí)上,ACP 提供的內(nèi)容遠(yuǎn)不止如此,它還有更多更高級(jí)的功能。

例如當(dāng)我們連接的 SFTP 服務(wù)器有多個(gè)時(shí),我們需要通過(guò)不同地址來(lái)獲得不同的連接對(duì)象。此時(shí)最笨的辦法是每個(gè)不同的地址,都復(fù)制多一份代碼,然后通過(guò)不同類的不同方法來(lái)實(shí)現(xiàn)。但這樣的情況工作量相當(dāng)可觀,并且也會(huì)有很多重復(fù)代碼。這種時(shí)候就可以使用 BaseKeyedPooledObjectFactory 來(lái)替代 BasePooledObjectFactory,從而實(shí)現(xiàn)通過(guò) key 來(lái)實(shí)現(xiàn)不同地址的連接對(duì)象管理。

更多關(guān)于 ACP 的內(nèi)容,感興趣的同學(xué)可以自行探索,這里就不深入講解了。

謝謝大家的閱讀。如果文章對(duì)你有幫助,點(diǎn)個(gè) 「點(diǎn)贊」 ,或者分享到朋友圈 吧。

參考資料

  • Apache Commons 系列簡(jiǎn)介 之 Pool-阿里云開(kāi)發(fā)者社區(qū)
  • Apache Common Pool2 對(duì)象池應(yīng)用淺析 - 知乎
  • Pool – Project Information

 本文轉(zhuǎn)載自微信公眾號(hào)「陳樹(shù)義」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系陳樹(shù)義公眾號(hào)。

 

責(zé)任編輯:武曉燕 來(lái)源: 陳樹(shù)義
相關(guān)推薦

2019-12-30 15:30:13

連接池請(qǐng)求PHP

2009-07-17 17:07:17

JDBC教程

2011-07-04 10:17:38

JDBC

2011-06-01 13:54:10

MySQL

2013-06-17 10:25:16

連接池Java

2022-03-09 09:43:01

工具類線程項(xiàng)目

2010-01-05 10:11:23

ADO.NET連接池

2018-02-07 16:23:58

連接池內(nèi)存池AI

2025-11-03 09:26:21

2022-11-11 09:41:04

連接池微服務(wù)數(shù)據(jù)庫(kù)

2009-06-17 16:22:45

Hibernate連接

2009-09-22 16:04:50

Hibernate連接

2009-09-22 14:52:55

Hibernate p

2010-11-08 16:46:57

2021-08-12 06:52:01

.NET數(shù)據(jù)庫(kù)連接池

2024-12-04 15:55:19

2009-06-24 07:53:47

Hibernate數(shù)據(jù)

2009-12-25 15:38:12

ADO連接池

2022-07-19 13:51:47

數(shù)據(jù)庫(kù)Hikari連接池

2018-06-19 16:04:27

Dubbo應(yīng)用Java
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

888av在线视频| 国产国产精品| 黄色大片在线免费看| 清纯唯美激情亚洲| 亚洲午夜黄色| 人人澡人人澡人人看欧美| 欧美日韩在线精品一区二区三区激情综合 | 国产美女精品在线| 精品视频高清无人区区二区三区| 日本三级韩国三级欧美三级| 国产91精品久久久久久| 91综合精品国产丝袜长腿久久| 久久99精品国产.久久久久久| 91九色国产视频| 亚洲精品一级二级三级| 影音先锋日韩有码| 国产精品99久久精品| 久久综合一区二区三区| 欧美大片1688| 亚洲国产精品va在线看黑人| 2019中文字幕在线视频| 欧美日韩一区二区三区四区五区| 日韩视频第一页| av免费看在线| 精品久久久久久久一区二区蜜臀| 国产又色又爽又黄刺激在线视频| 日韩欧美亚洲国产另类| 性欧美1819sex性高清大胸| 精品国产免费视频| 黄视频免费在线看| 亚洲精品资源在线| 国产精品伦一区二区| 久久国产精品亚洲| 一区二区三区视频免费观看 | 国产啊啊啊视频在线观看| 亚洲成在人线av| yy6080久久伦理一区二区| 欧美大片免费看| 日韩av在线播放网址| 成人免费视频网站| 视频精品一区二区| 精品人妻少妇一区二区| 欧美精彩视频一区二区三区| 免费特级黄毛片| 欧美精品乱码久久久久久按摩| 超碰高清在线| 97人人模人人爽人人喊中文字| 午夜国产精品视频| eeuss中文| 亚洲男人的天堂在线观看| 欧美高清电影在线| 亚洲级视频在线观看免费1级| 国产精品一区二区三区av| 国产女人精品视频| 久热成人在线视频| 99在线免费观看| 日韩精品一区二区在线| 国产精品一区免费在线 | 91国偷自产一区二区三区成为亚洲经典| 欧美jizz18hd性欧美| 在线播放国产一区中文字幕剧情欧美| 大奶在线精品| 国产乱码精品一区二区三区中文| 国产成人日日夜夜| 色资源网站在线观看| 日韩av网址在线| 欧美色爱综合| www插插插无码免费视频网站| 亚洲一区二区三区四区在线观看 | 自拍自偷一区二区三区| 国内精品久久国产| 成人免费毛片高清视频| 天堂中文资源在线| 中文字幕亚洲欧美在线| 影音先锋成人在线电影| 日韩精品视频久久| 欧美麻豆精品久久久久久| 麻豆视频久久| 欧美日韩在线高清| 亚洲免费色视频| 无遮挡爽大片在线观看视频| 国产精品久久久久久久久久99| 韩国成人福利片在线播放| 中文字幕在线观| 上原亚衣av一区二区三区| 欧美69wwwcom| 成人性生生活性生交12| 亚洲成人精品久久久| 97久久夜色精品国产| 少妇人妻互换不带套| 制服丝袜激情欧洲亚洲| 西野翔中文久久精品字幕| 一区二区不卡在线| 欧美特级限制片免费在线观看| 无码日韩精品一区二区免费| 阿v天堂2018| 日韩h在线观看| 久久国产毛片| 成人高清免费观看mv| 欧美亚洲成人xxx| 久久久精品2019中文字幕之3| 午夜久久中文| 午夜精品一区二区三区在线观看| 日韩欧美成人精品| 不卡视频在线| 日本中文字幕视频| 奇门遁甲1982国语版免费观看高清 | 欧美白嫩的18sex少妇| 中文字幕国产亚洲2019| 日韩精品一卡二卡三卡四卡无卡 | 成人欧美一区二区三区1314| av成人在线播放| 手机在线视频你懂的| 91精品国产aⅴ一区二区| 久久久久蜜桃| 全部a∨一极品视觉盛宴| 操人视频在线观看欧美| 国产精品一区在线| av影院在线免费观看| 蜜桃视频在线观看成人| 欧美在线观看视频在线| 欧美精品黄色| 91九色在线porn| 精品伦理一区二区三区| 欧美喷潮久久久xxxxx| 很黄很黄激情成人| 国产精品一区二区婷婷| 成人做爽爽免费视频| 福利一区福利二区微拍刺激| 天天射综合网视频| 久久手机免费观看| 亚洲在线视频观看| 欧美亚洲高清一区| 妖精视频成人观看www| 黄网站视频在线观看| 欧美极品色图| 亚洲国产精品免费| 国产福利91精品| 9999精品| 国产毛片毛片| 91影视免费在线观看| 在线观看欧美精品| 日本一不卡视频| 精品网站在线| mm131国产精品| 国产精品一区久久久| 在线观看日产精品| 美女任你摸久久 | 成人免费淫片在线费观看| 久久免费国产精品1| 亚洲免费观看高清在线观看| 日韩av二区| 在线免费黄色| 熟女熟妇伦久久影院毛片一区二区| 亚洲欧美精品伊人久久| 久久综合中文字幕| 少妇一区二区三区| 韩日在线视频| 一区二区三区精品国产| 久久中文字幕在线视频| 亚洲国产精品一区二区久久恐怖片| 午夜久久99| 欧美xx视频| 黄色毛片视频| 欧美日韩一区二区视频在线观看| 亚洲人成网站999久久久综合| 欧美国产精品v| 国内精品久久久久久久影视蜜臀 | 91动漫在线看| 欧美一级在线亚洲天堂| 欧美三级日本三级少妇99| 激情五月婷婷综合| 亚洲电影男人天堂| 中中文字幕av在线| 波多结衣在线观看| 国产青春久久久国产毛片| 中文字幕日韩精品有码视频| 亚洲国产精品人人做人人爽| 蜜臀av一区二区| 欧洲在线一区| 在线中文字幕视频观看| 91极品尤物在线播放国产| 精品人伦一区二区三区| 九九热视频这里只有精品| 91精品1区2区| 久久亚洲二区三区| 99精品热视频只有精品10| 国产一区二区三区视频在线| 992tv免费直播在线观看| 黄色片一级视频| 久久99精品久久久久久青青日本 | 国产福利久久精品| 色婷婷成人综合| 色欧美乱欧美15图片| 久久99蜜桃综合影院免费观看| 97视频在线观看视频免费视频 | 人妻熟女一二三区夜夜爱| 亚洲欧洲xxxx| 欧美日韩视频在线一区二区| 在线这里只有精品|