一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - java多線程編程之InheritableThreadLocal

java多線程編程之InheritableThreadLocal

2021-01-21 13:23WAUANG Java教程

這篇文章主要為大家詳細介紹了java多線程編程之InheritableThreadLocal,具有一定的參考價值,感興趣的小伙伴們可以參考一下

InheritableThreadLocal的作用: 當我們需要在子線程中使用父線程中的值得時候我們就可以像使用ThreadLocal那樣來使用InheritableThreadLocal了。

首先我們來看一下InheritableThreadLocal的jdk源碼:

java" id="highlighter_707595">
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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的關鍵所在,它提供了這個方法,返回父線程中的值,如果還需要在父線程上添加值則可以重寫childValue方法。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
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對繼承值進行修改

是不是有一個疑問,為什么子線程能獲取父線程的數據?

我們可以看到InheritableThreadLocal重寫了getMap方法和createMap方法,上一節講ThreadLocal的時候我們知道,ThreadLocal的值是存儲在一個叫ThreadLocals的變量中,但是現在返回一個InheritableThreadLocals,這個變量和ThreadLocals是一模一樣的只是名字換了,那么究竟 為什么在新的 線程中 通過 threadlocal.get() 方法還能得到值呢?

我們看childValue方法可以猜測到可能在線程創建的時候,做了一些手腳,做了一些值得傳遞。

我們打開Thread類的源碼的時候可以發現 :

ThreadLocal.ThreadLocalMap inheritableThreadLocals = null;

所以當我們創建一個子線程的時候,他就存在一個和ThreadLocals的一樣的InheritableThreadLocal變量,再往下看:

?
1
2
3
4
5
6
7
private void init(ThreadGroup g, Runnable target, String name,
           long stackSize, AccessControlContext acc,
           .
           .
           if (inheritThreadLocals && parent.inheritableThreadLocals != null)
      this.inheritableThreadLocals =
        ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);

重點是以下這段代碼:

?
1
2
3
if (inheritThreadLocals && parent.inheritableThreadLocals != null)
      this.inheritableThreadLocals =
        ThreadLocal.createInheritedMap(parent.inheritableThreadLocals);

繼續看:

?
1
2
3
static ThreadLocalMap createInheritedMap(ThreadLocalMap parentMap) {
   return new ThreadLocalMap(parentMap);
 }
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
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++;
          }
        }
      }
    }

有這段代碼,先得到父線程(也就是當前執行的線程)的值,然后用for循環一個個的將父線程中的值放入我們新創建的值中。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://blog.csdn.net/qq_39266910/article/details/78258498

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 青青国产成人久久激情91麻豆 | 国产一卡2卡3卡4卡公司科普 | 亚洲图片一区二区三区 | 91在线精品国产丝袜超清 | 成人中文字幕在线高清 | 大乳女子一级毛片 | 边吃奶边扎下面 | 亚洲好骚综合 | 好吊色青青青国产综合在线观看 | 狠狠色成人综合 | 邪恶肉肉全彩色无遮琉璃神社 | 99精品免费观看 | 国产欧美精品专区一区二区 | 深夜视频免费看 | 亚洲视频中文字幕 | 日韩在线成人 | 日韩一区在线播放 | 98精品全国免费观看视频 | 特黄特色大片免费视频大全 | jj视频免费看 | 华人在线视频 | 久久青草免费91线频观看站街 | 男人猛戳女人下部30分钟 | 顶级欧美做受xxx000 | 亚洲第一区二区快射影院 | 免费在线视频成人 | 百合文高h | 亚洲国产综合另类视频 | 青青国产成人久久91网 | 日本亚洲免费 | 火影忍者小南裸羞羞漫画 | 国产精品免费aⅴ片在线观看 | 性xxx欧美 | 91麻豆精品国产片在线观看 | 国产视频福利 | 青草视频网站 | 亚洲精品tv久久久久久久久久 | 国产在线视频第一页 | 6080伦理久久精品亚洲 | 婷婷丁香色综合狠狠色 | 全黄毛片 |