精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

讀懂HikariCP一百行代碼,多線程就是個(gè)孫子!

數(shù)據(jù)庫(kù) 其他數(shù)據(jù)庫(kù)
HikariCP是SpringBoot默認(rèn)的數(shù)據(jù)庫(kù)連接池,它毫不謙虛的的起了一個(gè)叫做光的名字,這讓國(guó)產(chǎn)Druid很沒面子。

總結(jié):Java屆很難得有讀百十行代碼就能增加修煉的機(jī)會(huì),這里有一個(gè)。

通常,我在看書的時(shí)候一般不寫代碼,因?yàn)槲业哪X袋被設(shè)定成單線程的,一旦同時(shí)喂給它不同的信息,它就無法處理。

但多線程對(duì)電腦來說就是小菜一碟,它可以同時(shí)做很多事,看起來匪夷所思。好希望把自己的大腦皮層移植到這些牛x的設(shè)備上。

用人腦思考電腦正在思考的問題,這本身就是一種折磨。但平常的工作和面試中,又不得不面對(duì)這樣的場(chǎng)景,所以多線程就成了編程路上一塊難啃的骨頭。

HikariCP是SpringBoot默認(rèn)的數(shù)據(jù)庫(kù)連接池,它毫不謙虛的的起了一個(gè)叫做光的名字,這讓國(guó)產(chǎn)Druid很沒面子。

還是言歸正傳,看一下Hikari中的ConcurrentBag吧。

核心數(shù)據(jù)結(jié)構(gòu)

多線程代碼一個(gè)讓人比較頭疼的問題,就是每個(gè)API我都懂,但就是不會(huì)用。很多對(duì)concurrent包倒背如流的同學(xué),在面對(duì)現(xiàn)實(shí)的問題時(shí),到最后依然不得不被迫加上Lock或者synchronized。

ConcurrentBag是一個(gè)Lock free的數(shù)據(jù)結(jié)構(gòu),主要用作數(shù)據(jù)庫(kù)連接的存儲(chǔ),可以說整個(gè)HikariCP的核心就是它。刪掉亂七八糟的注釋和異常處理,可以說關(guān)鍵的代碼也就百十來行,但里面的道道卻非常的多。

ConcurrentBag速度很快,要達(dá)到這個(gè)目標(biāo),就需要一定的核心數(shù)據(jù)結(jié)構(gòu)支持。

private final CopyOnWriteArrayList<T> sharedList;
private final ThreadLocal<List<Object>> threadList;
private final AtomicInteger waiters;
private final SynchronousQueue<T> handoffQueue;
  • sharedList 用來緩存所有的連接,是一個(gè)CopyOnWriteArrayList結(jié)構(gòu)。
  • threadList 用來緩存某個(gè)線程所使用的所有連接,相當(dāng)于快速引用,是一個(gè)ThreadLocal類型的ArrayList。
  • waiters 當(dāng)前正在獲取連接的等待者數(shù)量。AtomicInteger,就是一個(gè)自增對(duì)象。當(dāng)waiters的數(shù)量大于0時(shí)候,意味著有線程正在獲取資源。
  • handoffQueue 0容量的快速傳遞隊(duì)列,SynchronousQueue類型的隊(duì)列,非常有用。

ConcurrentBag里面的元素,為了能夠無鎖化操作,需要使用一些變量來標(biāo)識(shí)現(xiàn)在處于的狀態(tài)。抽象的接口如下:

public interface IConcurrentBagEntry{
int STATE_NOT_IN_USE = 0;
int STATE_IN_USE = 1;
int STATE_REMOVED = -1;
int STATE_RESERVED = -2;

boolean compareAndSet(int expectState, int newState);
void setState(int newState);
int getState();
}

有了這些數(shù)據(jù)結(jié)構(gòu)的支持,我們的ConcurrentBag就可以實(shí)現(xiàn)它光的宣稱了。

獲取連接

連接的獲取是borrow方法,還可以傳入一個(gè)timeout作為超時(shí)控制。

public T borrow(long timeout, final TimeUnit timeUnit) throws InterruptedException

首先,如果某個(gè)線程執(zhí)行非常快,使用了比較多的連接,就可以使用ThreadLocal的方式快速獲取連接對(duì)象,而不用跑到大池子里面去獲取。代碼如下。

// Try the thread-local list first
final var list = threadList.get();
for (int i = list.size() - 1; i >= 0; i--) {
final var entry = list.remove(i);
final T bagEntry = weakThreadLocals ? ((WeakReference<T>) entry).get() : (T) entry;
if (bagEntry != null && bagEntry.compareAndSet(STATE_NOT_IN_USE, STATE_IN_USE)) {
return bagEntry;
}
}

我們都知道,包括ArrayList和HashMap一些基礎(chǔ)的結(jié)構(gòu),都是Fail Fast的,如果你在遍歷的時(shí)候,刪掉一些數(shù)據(jù),有可能會(huì)引起問題。幸運(yùn)的是,由于我們的List是從ThreadLocal獲取的,它首先就避免了線程安全的問題。

接下來就是遍歷。這段代碼采用的是尾遍歷(頭遍歷會(huì)出現(xiàn)錯(cuò)誤),用于快速的從列表中找到一個(gè)可以復(fù)用的對(duì)象,然后使用CAS來把狀態(tài)置為使用中。但如果對(duì)象正在被使用,則直接刪除它。

在ConcurrentBag里,每個(gè)ThreadLocal最多緩存50個(gè)連接對(duì)象引用。

當(dāng)ThreadLocal里找不到可復(fù)用的對(duì)象,它就會(huì)到大池子里去拿。也就是下面這段代碼。

// Otherwise, scan the shared list ... then poll the handoff queue
final int waiting = waiters.incrementAndGet();
try {
for (T bagEntry : sharedList) {
if (bagEntry.compareAndSet(STATE_NOT_IN_USE, STATE_IN_USE)) {
// If we may have stolen another waiter's connection, request another bag add.
if (waiting > 1) {
listener.addBagItem(waiting - 1);
}
return bagEntry;
}
}

listener.addBagItem(waiting);

// 還拿不到,就需要等待別人釋放了
timeout = timeUnit.toNanos(timeout);
do {
final var start = currentTime();
final T bagEntry = handoffQueue.poll(timeout, NANOSECONDS);
if (bagEntry == null || bagEntry.compareAndSet(STATE_NOT_IN_USE, STATE_IN_USE)) {
return bagEntry;
}

timeout -= elapsedNanos(start);
} while (timeout > 10_000);

return null;
}
finally {
waiters.decrementAndGet();
}

首先要注意,這段代碼可能是由不同的線程執(zhí)行的,所以必須要考慮線程安全問題。由于shardList是線程安全的CopyOnWriteArrayList,適合讀多寫少的場(chǎng)景,我們可以直接進(jìn)行遍歷。

這段代碼的目的是一樣的,需要從sharedList找到一個(gè)空閑的連接對(duì)象。這里把自增的waiting變量傳遞到外面的代碼進(jìn)行處理,主要是由于想要根據(jù)waiting的大小來確定是否創(chuàng)建新的對(duì)象。

如果無法從池子里獲取連接,則需要等待別的線程釋放一些資源。

創(chuàng)建對(duì)象的過程是異步的,要想獲取它,還需要依賴一段循環(huán)代碼。while循環(huán)代碼是納秒精度,會(huì)嘗試從handoffQueue里獲取。最終會(huì)調(diào)用SynchronousQueue的transfer方法。

歸還連接

有借就有還,當(dāng)某個(gè)連接使用完畢,它將被歸還到池子中。

public void requite(final T bagEntry)
{
bagEntry.setState(STATE_NOT_IN_USE);

for (var i = 0; waiters.get() > 0; i++) {
if (bagEntry.getState() != STATE_NOT_IN_USE || handoffQueue.offer(bagEntry)) {
return;
}
else if ((i & 0xff) == 0xff) {
parkNanos(MICROSECONDS.toNanos(10));
}
else {
Thread.yield();
}
}

final var threadLocalList = threadList.get();
if (threadLocalList.size() < 50) {
threadLocalList.add(weakThreadLocals ? new WeakReference<>(bagEntry) : bagEntry);
}
}

首先,把這個(gè)對(duì)象置為可用狀態(tài)。然后,代碼會(huì)進(jìn)入一個(gè)循環(huán),等待使用方把這個(gè)連接接手過去。當(dāng)連接處于STATE_NOT_IN_USE狀態(tài),或者隊(duì)列中的數(shù)據(jù)被取走了,那么就可以直接返回了。

由于waiters.get()是實(shí)時(shí)獲取的,有可能長(zhǎng)時(shí)間一直大于0,這樣代碼就會(huì)變成死循環(huán),浪費(fèi)CPU。代碼會(huì)嘗試不同層次的睡眠,一個(gè)是每隔255個(gè)waiter睡10ns,一個(gè)是使用yield讓出cpu時(shí)間片。

如果歸還連接的時(shí)候并沒有被其他線程獲取到,那么最后我們會(huì)把歸還的連接放入到相對(duì)應(yīng)的ThreadLocal里,因?yàn)閷?duì)一個(gè)連接來說,借和還,通常是一個(gè)線程。

知識(shí)點(diǎn)

看起來平平無奇的幾行代碼,為什么搞懂了就能Hold住大部分的并發(fā)編程場(chǎng)景呢?主要還是這里面的知識(shí)點(diǎn)太多。下面我簡(jiǎn)單羅列一下,你可以逐個(gè)攻破。

  • 使用ThreadLocal來緩存本地資源引用,使用線程封閉的資源來減少鎖的沖突。
  • 采用讀多寫少的線程安全的CopyOnWriteArrayList來緩存所有對(duì)象,幾乎不影響讀取效率。
  • 使用基于CAS的AtomicInteger來計(jì)算等待者的數(shù)量,無鎖操作使得計(jì)算更加快速。
  • 0容量的交換隊(duì)列SynchronousQueue,使得對(duì)象傳遞更加迅速。
  • 采用compareAndSet的CAS原語來控制狀態(tài)的變更,安全且效率高。很多核心代碼都是這么設(shè)計(jì)的。
  • 在循環(huán)中使用park、yield等方法,避免死循環(huán)占用大量CPU。
  • 需要了解并發(fā)數(shù)據(jù)結(jié)構(gòu)中的offer、poll、peek、put、take、add、remove方法的區(qū)別,并靈活應(yīng)用。
  • CAS在設(shè)置狀態(tài)時(shí),采用了volatile關(guān)鍵字修飾,對(duì)于volatile的使用也是一個(gè)常見的優(yōu)化點(diǎn)。
  • 需要了解WeakReference弱引用在垃圾回收時(shí)候的表現(xiàn)。

麻雀雖小,五臟俱全。如果你想要你的多線程編程能力更上一層樓,讀一讀這個(gè)短小精悍的ConcurrentBag吧。當(dāng)你掌握了它,多線程的那些東西,不過是小菜一碟。

作者簡(jiǎn)介:小姐姐味道  (xjjdog),一個(gè)不允許程序員走彎路的公眾號(hào)。聚焦基礎(chǔ)架構(gòu)和Linux。十年架構(gòu),日百億流量,與你探討高并發(fā)世界,給你不一樣的味道。

責(zé)任編輯:武曉燕 來源: 小姐姐味道
相關(guān)推薦

2020-03-12 10:16:45

代碼Java多線程

2015-09-14 09:07:15

Java多線程

2023-12-07 07:28:25

線程共享資源

2019-09-24 14:19:12

PythonC語言文章

2019-12-04 10:20:57

代碼開發(fā)工具

2020-08-12 09:53:18

代碼開發(fā)工具

2010-02-04 10:19:39

C++多線程

2012-01-12 10:09:30

Java

2020-06-16 11:00:40

線程Java代碼

2015-12-22 10:39:52

Java多線程問題

2017-01-19 10:24:38

Java多線程問題

2024-10-17 09:29:06

2025-07-11 00:57:30

2009-07-17 17:29:13

多任務(wù)多線程

2010-03-17 19:24:38

Java多線程循環(huán)

2009-03-12 10:52:43

Java線程多線程

2020-04-07 11:10:30

Python數(shù)據(jù)線程

2010-01-08 10:48:05

VB.NET多線程

2010-03-17 09:33:30

Java多線程方案

2023-06-05 07:56:10

線程分配處理器
點(diǎn)贊
收藏

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

国产精品乱看| 四虎精品在线观看| 91免费观看在线| 国产精品久久久久久久久久免费| 亚洲精品国产熟女久久久| 天然素人一区二区视频| 最新热久久免费视频| 国产精品jizz视频| 无码人妻av一区二区三区波多野| 99精品电影| 日韩电视剧在线观看免费网站| 人妻内射一区二区在线视频 | 欧美三级一区二区三区| 成人女性视频| 精品国产一区二区三区忘忧草 | 99久久er| 婷婷成人激情在线网| 亚洲日本欧美在线| 无码国产精品96久久久久| 日产国产高清一区二区三区| 欧美人在线观看| 欧美a在线播放| 久久香蕉网站| 日韩精品一区二区三区中文不卡| 妺妺窝人体色www在线观看| 都市激情久久综合| 国产精品女人毛片| 欧美日韩一区二区三区在线视频| 亚洲成人777777| 久久国产精品露脸对白| 日本中文字幕不卡免费| 久久夜色精品亚洲| 国语精品一区| 另类少妇人与禽zozz0性伦| 人妻体内射精一区二区| 伦理一区二区三区| 精品久久五月天| 超碰中文字幕在线观看| 精品乱码一区二区三区四区| 狠狠色狠狠色综合日日五| 天堂а√在线中文在线| 成人影院在线观看| 亚洲欧洲av色图| 亚洲欧洲一区二区福利| 精品视频一二三| 久久久综合视频| 久久天堂国产精品| 污污视频在线免费看| av网站一区二区三区| 高清一区二区三区视频| www五月婷婷| 国产成人啪免费观看软件| 91在线视频九色| 国产精品伦理一区| 狠狠网亚洲精品| 91久久精品美女高潮| 国产精品久久久久久无人区| 捆绑调教一区二区三区| 国产色婷婷国产综合在线理论片a| 日本精品入口免费视频| 日韩不卡一二三区| 国产精品入口免费视频一| 亚洲性猛交富婆| 美女mm1313爽爽久久久蜜臀| 国产精品网红直播| 一级片在线免费观看视频| 久久国产精品一区二区| 91在线观看免费网站| www香蕉视频| 99久久久免费精品国产一区二区| 久久爱av电影| 国产午夜在线视频| 中文字幕亚洲一区二区va在线| 91免费网站视频| 牛牛电影国产一区二区| 亚洲.国产.中文慕字在线| 久久无码高潮喷水| 高清在线一区| 日韩欧美电影一区| 亚洲制服丝袜在线播放| 国产videos久久| 久久夜色精品国产亚洲aⅴ| 久久精品波多野结衣| 一区二区日韩免费看| 国产精品99导航| 国产精品永久久久久久久久久| 黑人巨大精品欧美一区| 国产成人一区二区三区免费看| 亚洲人成色777777精品音频| 91女厕偷拍女厕偷拍高清| 久久这里只有精品国产| 欧美专区一区| 亚洲国产精品高清久久久| 女同毛片一区二区三区| 欧美一区二区三区激情视频| 久久精品视频亚洲| 国产精品第一页在线观看| 亚洲尤物在线| 18视频免费网址在线观看| 日本成人中文字幕在线视频| 成人激情在线观看| 理论片中文字幕| 国产无一区二区| av中文字幕av| 亚洲少妇视频| 欧美一区欧美二区| 女同毛片一区二区三区| 亚洲国产精品成人| 日韩免费视频在线观看| 精品国自产在线观看| 久久综合丝袜日本网| 裸体裸乳免费看| 成人国产精品一级毛片视频| 欧美性高潮床叫视频| 污视频网站观看| 巨人精品**| 国产91精品久| v天堂中文在线| 欧美三级伦理在线| 久久久久久久一区二区三区| 亚洲熟妇无码另类久久久| 波多野结衣视频一区二区| 欧美猛男男办公室激情| 亚洲麻豆一区二区三区| 国产精品成人一区二区不卡| 91成人福利在线| 亚洲第一页在线观看| 国产精品久久久久久久久免费丝袜| 男女超爽视频免费播放| 国产高清日韩| 日韩中文在线观看| 伊人成年综合网| 91啦中文在线观看| 国产婷婷一区二区三区| 97青娱国产盛宴精品视频| 美女999久久久精品视频| 日本三级一区二区三区| 26uuu亚洲综合色| 国产精品国产亚洲精品看不卡| 1204国产成人精品视频| 久久综合免费视频影院| 97成人免费视频| 中文字幕高清不卡| 色一情一乱一伦一区二区三区日本| 欧美影院天天5g天天爽| 97人洗澡人人免费公开视频碰碰碰| 精品人妻伦一区二区三区久久| 中文字幕制服丝袜成人av| 无码日韩人妻精品久久蜜桃| 国产成人三级| 国产精品99久久久久久人| 黄色片视频在线观看| 一本大道av伊人久久综合| 中文字幕5566| 日韩二区三区四区| 色噜噜狠狠一区二区三区| 欧美aaa视频| 最近中文字幕日韩精品| 91亚洲欧美激情| 亚洲免费观看高清完整| 日本黄色一级网站| 亚洲性感美女99在线| 国产伦精品一区二区三毛| 大桥未久在线视频| 亚洲欧美中文日韩在线| 波多野结衣一区二区三区在线| 国产亚洲成年网址在线观看| 尤蜜粉嫩av国产一区二区三区| 日韩大片在线播放| 亚洲一区二区少妇| 欧美24videosex性欧美| 亚洲黄色在线看| 日批视频免费在线观看| 国产精品久久久久久亚洲毛片| 国产美女18xxxx免费视频| 亚洲色图网站| 国产一区二区不卡视频在线观看| 亚洲深夜视频| 色综合伊人色综合网| www黄色在线观看| 欧美性猛交xxxx偷拍洗澡| 日本理论中文字幕| 国产精品自拍在线| 国产精品一区二区免费在线观看| 女厕嘘嘘一区二区在线播放| 国产欧美一区二区三区视频 | 成人小视频免费观看| 欧美国产激情视频| 97精品国产福利一区二区三区| www.久久艹| 美女100%一区| 久久综合五月天| 日本一卡二卡四卡精品| 欧美日本一区二区| 国产成人在线免费观看视频| 国产精品三级视频| 国产情侣久久久久aⅴ免费| 日一区二区三区| 97av中文字幕| 成人精品影院| 久久久久久久久久久一区 | 欧美激情一区二区三区蜜桃视频| 91九色露脸| 午夜影视一区二区三区| 日韩在线观看免费网站| 亚洲免费不卡视频| 欧美色倩网站大全免费| 久久久久久久九九九九| 国产日韩v精品一区二区| 国产精品19p| 三区四区电影在线观看| 日韩三级精品电影久久久| 天天干天天干天天| 亚洲精品久久7777| 五月婷婷欧美激情| 91小视频在线免费看| www.日本久久| 奇米精品一区二区三区四区 | 91九色在线观看| 精品美女一区| 国产成人一区二区三区小说| jizz一区二区三区| 另类视频在线观看| 蜜桃视频网站在线观看| 国产一区二区三区18 | 中文字幕亚洲欧美| 性xxxx视频播放免费| 日韩欧美国产三级电影视频| 中文字幕在线观看欧美| 色婷婷综合久久久| 久久露脸国语精品国产91| 亚洲伊人伊色伊影伊综合网| 538精品在线视频| 国产精品高清亚洲| 性爱在线免费视频| 国产欧美日本一区视频| av直播在线观看| 91在线精品秘密一区二区| 人妻互换一二三区激情视频| 国产精品一区免费在线观看| 色婷婷激情视频| 国内精品自线一区二区三区视频| 日本人视频jizz页码69| 日韩电影在线一区| 一本久道中文无码字幕av| 久久九九精品| 丁香婷婷激情网| 美女视频黄久久| 女同激情久久av久久| 久久99久久99小草精品免视看| 亚洲精品久久久中文字幕| 美美哒免费高清在线观看视频一区二区| 国产精品人人爽人人爽| 免费国产亚洲视频| 亚洲综合在线一区二区| 国产成人av电影在线播放| 一二三区视频在线观看| 粉嫩高潮美女一区二区三区| 午夜不卡久久精品无码免费| 91最新地址在线播放| 亚洲综合色一区| 国产精品入口麻豆原神| 亚洲天堂一级片| 亚洲一区二区视频| 特一级黄色大片| 在线国产电影不卡| 一级片aaaa| 精品久久一区二区三区| 欧美欧美欧美| www.欧美免费| 日日骚久久av| 国产哺乳奶水91在线播放| 日韩欧美成人激情| 日批免费在线观看| 亚洲欧美日韩久久久久久 | 色999日韩欧美国产| 在线中文字幕视频观看| 97婷婷涩涩精品一区| 蜜桃精品在线| 国产精品揄拍500视频| 精品午夜av| 欧美极品色图| 久久久久av| 男人天堂1024| 精品中文字幕一区二区小辣椒| xxxx国产视频| 国产日本一区二区| 91视频综合网| 91久久线看在观草草青青| 亚洲天堂手机版| 亚洲国产精久久久久久久| 成人午夜在线观看视频| 欧美精品www| 日韩一区精品| www.久久草| 色综合久久一区二区三区| 日韩亚洲欧美视频| 青青青爽久久午夜综合久久午夜| 中文字幕在线国产| 国产精品日日摸夜夜摸av| 日本熟妇毛耸耸xxxxxx| 欧美三级电影网站| 天天干,夜夜爽| 久久久国产一区二区| 欧美91看片特黄aaaa| av电影成人| 日韩免费高清| 国产精品沙发午睡系列| 国产精品99久久久久久久女警| 四虎国产精品成人免费入口| 亚洲一区二区在线观看视频| 91精品国产乱码久久| 亚洲精品一区二区久| 新版中文在线官网| 国产日韩中文在线| 精品欧美激情在线观看| 亚洲人精品午夜射精日韩| 国内成人精品2018免费看| 国产jk精品白丝av在线观看 | 中文字幕人妻精品一区| 亚洲精品二三区| 亚洲小说区图片| 91在线直播亚洲| 天天综合一区| 中文字幕 91| 国产丝袜美腿一区二区三区| 国产女同在线观看| 欧美精品一区视频| 在线播放蜜桃麻豆| 91亚洲国产精品| 91九色精品| 日韩一区二区三区久久| 国产欧美精品国产国产专区| 中文字幕在线欧美| 精品调教chinesegay| 国产美女高潮在线观看| 国产精品一区二区av| 国产精品观看| yjizz视频| 亚洲成人av一区二区三区| 粉嫩av一区二区夜夜嗨| 国外成人免费在线播放| 国内精品国产成人国产三级粉色 | 污污的网站在线免费观看| 91精品视频在线看| 亚洲精品成人| 色婷婷一区二区三区在线观看| 最新热久久免费视频| 99热这里只有精品在线观看| 免费99精品国产自在在线| 国产精品99久久免费| 国产欧美综合一区| 国产精一品亚洲二区在线视频| 欧美日韩偷拍视频| 精品国产一区二区三区av性色 | 香蕉久久网站| 99999精品| 亚洲福中文字幕伊人影院| 少妇人妻一区二区| 2020欧美日韩在线视频| 九九在线精品| 久久婷五月综合| 亚洲柠檬福利资源导航| 人妻与黑人一区二区三区| 国产91精品久久久久久久| 国产成人ay| 久久久久久久久久久久久久久国产 | 激情亚洲网站| 国产精品jizz| 欧美日韩日本视频| 性爱视频在线播放| 久久影院理伦片| 蜜臀av性久久久久蜜臀aⅴ流畅 | 亚洲免费激情| 亚洲a v网站| 欧美精品1区2区| 国产黄色大片在线观看| 精品一区二区三区自拍图片区| 日韩电影在线一区| 九九热国产在线| 亚洲精品中文字幕有码专区| www.成人在线视频| www国产免费| 国产三级一区二区三区| 国产成人精品毛片| 日本精品视频在线观看| 亚洲色图二区| 日韩乱码人妻无码中文字幕久久| 欧美蜜桃一区二区三区| 牛牛精品一区二区| 伊人久久大香线蕉精品| 成人深夜在线观看| 中国女人真人一级毛片| 欧美激情a在线| 欧美最新另类人妖| 岛国精品一区二区三区| 在线观看国产一区二区| wwww亚洲| 中国人体摄影一区二区三区| 97se亚洲国产综合自在线观|