Hystrix系列之ThreadLocal跨線程傳遞問題

在Hystrix系列之前的文章中提到過,如果使用線程池模式,那么存在一個(gè)ThreadLocal變量跨線程傳遞的問題,即在主線程的ThreadLocal變量,無法在線程池中使用,不過Hystrix內(nèi)部提供了解決方案,但是個(gè)人覺得這個(gè)方案不是那么友好。

解決方案

在Hystrix中,如果想在跨線程時(shí)共享數(shù)據(jù),必須通過HystrixRequestVariableDefault申明變量

HystrixRequestVariableDefault name = new HystrixRequestVariableDefault();
name.set("占小狼");

其實(shí)在用法上,和ThreadLocal是一樣的,只是需要對(duì)現(xiàn)有代碼的大量改造。

看下這個(gè)方案的實(shí)現(xiàn)原理,先從set開始。

public void set(T value) {
  HystrixRequestContext.getContextForCurrentThread().state.put(this, new LazyInitializer<T>(this, value));
}

Hystrix內(nèi)部通過HystrixRequestContext實(shí)現(xiàn)數(shù)據(jù)的跨線程傳遞,getContextForCurrentThread得到的是當(dāng)前線程的HystrixRequestContext對(duì)象,每個(gè)HystrixRequestContext對(duì)象都有一個(gè)對(duì)應(yīng)ConcurrentHashMap變量state,負(fù)責(zé)保存通過HystrixRequestVariableDefault初始化的數(shù)據(jù)。

通過set方法,該對(duì)象和數(shù)據(jù)會(huì)被保存在一個(gè)當(dāng)前線程所屬的map中。為了實(shí)現(xiàn)數(shù)據(jù)的跨線程傳遞,只需要在初始化task的時(shí)候,把主線程的HystrixRequestContext變量保存起來,在task執(zhí)行的時(shí)候,重新賦值到子線程的上下文中,這樣在子線程中就可以順利拿到這些數(shù)據(jù)。

Hystrix中通過HystrixContextCallable包裝原始Callable,并使用parentThreadState保存了當(dāng)前線程的HystrixRequestContext變量。

任務(wù)執(zhí)行時(shí),先保存子線程現(xiàn)有的HystrixRequestContext變量,再賦值主線程的HystrixRequestContext變量,任務(wù)執(zhí)行完成后,重新還原子線程。

如果不想使用Hystrix這種方式實(shí)現(xiàn),也可以使用Hystrix提供的插件方式重新包裝task,通過實(shí)現(xiàn)HystrixConcurrencyStrategy類,重寫wrapCallable方法,和Hystrix的實(shí)現(xiàn)原理類似。

比如提供一個(gè)繼承ThreadLocalXXXThreadLocal類,那么業(yè)務(wù)方在使用時(shí),就可以這樣使用。

ThreadLocal name = new XXXThreadLocal();
name.set("占小狼");

這種方式看起來對(duì)已有邏輯只有一點(diǎn)小小的改動(dòng),對(duì)于新接入的也不那么陌生。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,590評(píng)論 19 139
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,111評(píng)論 25 709
  • 人與人之間,如果能夠非暴力溝通,就能減少矛盾、沖突和爭(zhēng)執(zhí),從而與他人和諧相處,獲得良好的人際關(guān)系。這里給大家...
    十里晴晝閱讀 464評(píng)論 3 14
  • 一、從何而來 易澤考研,是易澤教育旗下針對(duì)考研領(lǐng)域的教育服務(wù)項(xiàng)目。團(tuán)隊(duì)創(chuàng)始人分別來自于清華大學(xué)、北京大學(xué)、南開大學(xué)...
    優(yōu)研優(yōu)選閱讀 365評(píng)論 0 0
  • 過去的童年不是那么好所以現(xiàn)在不管怎樣都不會(huì)說不好 過了好多好多的不是日子的日子想想都覺得害怕 我真的很佩服自己是那...
    統(tǒng)有你知足閱讀 245評(píng)論 0 0

友情鏈接更多精彩內(nèi)容