java多線程編程之InheritableThreadLocal
InheritableThreadLocal的作用: 當(dāng)我們需要在子線程中使用父線程中的值得時候我們就可以像使用ThreadLocal那樣來使用InheritableThreadLocal了。
首先我們來看一下InheritableThreadLocal的jdk源碼:
package java.lang;
import java.lang.ref.*;
public class InheritableThreadLocal<T> extends ThreadLocal<T> {
protected T childValue(T parentValue) {
return parentValue;
}
ThreadLocalMap getMap(Thread t) {
return t.inheritableThreadLocals;
}
void createMap(Thread t, T firstValue) {
t.inheritableThreadLocals = new ThreadLocalMap(this, firstValue);
}
}
這段代碼就是InheritableThreadLocal的完整源碼(刪除了很長的注釋)。
首先我們可以看到它是繼承ThreadLocal類的,然后提供了:
protected T childValue(T parentValue){}方法,這就是InheritableThreadLocal的關(guān)鍵所在,它提供了這個方法,返回父線程中的值,如果還需要在父線程上添加值則可以重寫childValue方法。
package InheritableThreadLocal;
import java.util.Date;
public class InheritableThreadLocaExt extends InheritableThreadLocal{
protected Object initialValue() {
return new Date().getTime();
}
protected Object childValue(Object parentValue) {
return parentValue+"對繼承值進行修改";
}
}
package InheritableThreadLocal;
public class tool {
public static InheritableThreadLocaExt t=new InheritableThreadLocaExt();
}
package InheritableThreadLocal;
public class MyThread extends Thread{
public void run() {
try {
for(int i=0;i<10;i++) {
System.out.println("在線程A中:"+tool.t.get());
sleep(100);
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
package InheritableThreadLocal;
public class test {
public static void main(String[] args) {
try {
for(int i=0;i<10;i++) {
System.out.println("主線程中值:"+tool.t.get());
Thread.sleep(100);
}
Thread.sleep(5000);
MyThread thread=new MyThread();
thread.start();
}catch(InterruptedException e){
e.printStackTrace();
}
}
}
運行輸出:
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
主線程中值:1508210392057
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
在線程A中:1508210392057對繼承值進行修改
是不是有一個疑問,為什么子線程能獲取父線程的數(shù)據(jù)?
我們可以看到InheritableThreadLocal重寫了getMap方法和createMap方法,上一節(jié)講ThreadLocal的時候我們知道,ThreadLocal的值是存儲在一個叫ThreadLocals的變量中,但是現(xiàn)在返回一個InheritableThreadLocals,這個變量和ThreadLocals是一模一樣的只是名字換了,那么究竟 為什么在新的 線程中 通過 threadlocal.get() 方法還能得到值呢?
我們看childValue方法可以猜測到可能在線程創(chuàng)建的時候,做了一些手腳,做了一些值得傳遞。
我們打開Thread類的源碼的時候可以發(fā)現(xiàn) :
ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;
所以當(dāng)我們創(chuàng)建一個子線程的時候,他就存在一個和ThreadLocals的一樣的InheritableThreadLocal變量,再往下看:
private void init(ThreadGroup g, Runnable target, String name,
long stackSize, AccessControlContext acc,
.
.
if (inheritThreadLocals && parent.inheritableThreadLocals != null)
this.inheritableThreadLocals =
ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);
重點是以下這段代碼:
if (inheritThreadLocals && parent.inheritableThreadLocals != null)
this.inheritableThreadLocals =
ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);
繼續(xù)看:
static ThreadLocalMap createInheritedMap(ThreadLocalMap parentMap) {
return new ThreadLocalMap(parentMap);
}
private ThreadLocalMap(ThreadLocalMap parentMap) {
Entry[] parentTable = parentMap.table;
int len = parentTable.length;
setThreshold(len);
table = new Entry[len];
for (int j = 0; j < len; j++) {
Entry e = parentTable[j];
if (e != null) {
@SuppressWarnings("unchecked")
ThreadLocal<Object> key = (ThreadLocal<Object>) e.get();
if (key != null) {
Object value = key.childValue(e.value);
Entry c = new Entry(key, value);
int h = key.threadLocalHashCode & (len - 1);
while (table[h] != null)
h = nextIndex(h, len);
table[h] = c;
size++;
}
}
}
}
有這段代碼,先得到父線程(也就是當(dāng)前執(zhí)行的線程)的值,然后用for循環(huán)一個個的將父線程中的值放入我們新創(chuàng)建的值中。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
一種新的日期處理方式之JavaScript Temporal API
JavaScript Temporal API是一種為Web開發(fā)人員提供了一種新的處理日期和時間數(shù)據(jù)類型的方式。它的目的是使操作日期和時間更加簡單和可靠,而且不用擔(dān)心歷史時區(qū)問題或全球化協(xié)調(diào)時間(UTC)之類的問題,感興趣的同學(xué)可以參考閱讀2023-05-05
解決spring-boot-starter-web等報紅問題
這篇文章主要介紹了解決spring-boot-starter-web等報紅問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-07-07
Intellij IDEA基于Springboot的遠程調(diào)試(圖文)
這篇文章主要介紹了Intellij IDEA基于Springboot的遠程調(diào)試(圖文),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10
如何獲取springboot打成jar后的classpath
這篇文章主要介紹了如何獲取springboot打成jar后的classpath問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-07-07
Java訪問WebService返回XML數(shù)據(jù)的方法
這篇文章主要介紹了Java訪問WebService返回XML數(shù)據(jù)的方法,涉及java操作WebService的相關(guān)技巧,需要的朋友可以參考下2015-06-06
SpringSecurity中的Filter Chain(過濾器鏈)
Spring Security的Filter Chain是由一系列過濾器組成的管道,每個過濾器執(zhí)行特定的安全功能,Spring Security能夠提供強大而靈活的安全控制機制,從而保護你的應(yīng)用程序不受各種網(wǎng)絡(luò)安全威脅的侵害,本文介紹SpringSecurity中的Filter Chain,感興趣的朋友跟隨小編一起看看吧2024-06-06

