欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java多線程編程之InheritableThreadLocal

 更新時間:2017年10月18日 10:59:06   作者:WAUANG  
這篇文章主要為大家詳細(xì)介紹了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)文章

  • Java中FilterInputStream和FilterOutputStream的用法詳解

    Java中FilterInputStream和FilterOutputStream的用法詳解

    這篇文章主要介紹了Java中FilterInputStream和FilterOutputStream的用法詳解,這兩個類分別用于封裝輸入和輸出流,需要的朋友可以參考下
    2016-06-06
  • JAVA 多線程之信號量(Semaphore)實例詳解

    JAVA 多線程之信號量(Semaphore)實例詳解

    這篇文章主要介紹了JAVA 多線程之信號量(Semaphore)實例詳解的相關(guān)資料,需要的朋友可以參考下
    2017-01-01
  • 一篇文章帶你理解Java的SPI機制(圖文并茂)

    一篇文章帶你理解Java的SPI機制(圖文并茂)

    本文詳細(xì)介紹了Java的SPI機制,包括其定義、用途和實現(xiàn)方式,SPI(ServiceProviderInterface)是一種服務(wù)發(fā)現(xiàn)機制,用于實現(xiàn)框架或庫的擴展點,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-10-10
  • MapStruct到底是什么?

    MapStruct到底是什么?

    今天給大家?guī)淼氖顷P(guān)于Java的相關(guān)知識,文中圍繞MapStruct到底是什么展開,文中有非常詳細(xì)的解釋及代碼示例,需要的朋友可以參考下
    2021-06-06
  • Xml中使用foreach遍歷對象實現(xiàn)代碼

    Xml中使用foreach遍歷對象實現(xiàn)代碼

    這篇文章主要介紹了Xml中使用foreach遍歷對象實現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-12-12
  • Java中的反射機制示例詳解

    Java中的反射機制示例詳解

    反射就是把Java類中的各個成分映射成一個個的Java對象。本文將通過示例詳細(xì)講解Java中的反射機制,感興趣的小伙伴可以跟隨小編學(xué)習(xí)一下
    2022-03-03
  • 解決IDEA中快捷鍵Alt+Enter不能使用的問題

    解決IDEA中快捷鍵Alt+Enter不能使用的問題

    這篇文章主要介紹了解決IDEA中快捷鍵Alt+Enter不能使用的問題,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-12-12
  • mybatis中resultMap 標(biāo)簽的使用教程

    mybatis中resultMap 標(biāo)簽的使用教程

    resultMap 標(biāo)簽用來描述如何從數(shù)據(jù)庫結(jié)果集中來加載對象,這篇文章重點給大家介紹mybatis中resultMap 標(biāo)簽的使用,感興趣的朋友一起看看吧
    2018-07-07
  • jmeter正則表達式實例詳解

    jmeter正則表達式實例詳解

    正則表達式就是記錄文本規(guī)則的代碼。學(xué)習(xí)正則表達式最好就是從實例下手。下面我們通過實例代碼給大家介紹jmeter正則表達式的相關(guān)知識,感興趣的朋友一起看看吧
    2021-12-12
  • java簡單實現(xiàn)斗地主發(fā)牌功能

    java簡單實現(xiàn)斗地主發(fā)牌功能

    這篇文章主要為大家詳細(xì)介紹了java簡單實現(xiàn)斗地主發(fā)牌功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06

最新評論