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

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

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

服務(wù)器之家 - 編程語言 - Java教程 - java 中ThreadLocal實例分析

java 中ThreadLocal實例分析

2020-11-17 11:56jamesjxin Java教程

這篇文章主要介紹了java 中ThreadLocal實例分析的相關(guān)資料,需要的朋友可以參考下

java  中ThreadLocal實例分析

從概念上理解,threadlocal使變量在多個線程中相互隔離實現(xiàn)線程安全,threadlocal包裝的變量最終都專屬于對應(yīng)的每個線程,線程之間相互獨立,用一個具體實現(xiàn)來說明:

?
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
public interface Consumer {
  int consume();
}
public class ComsumeThread implements Runnable {
 
  private Consumer consumer;
 
  public ComsumeThread(Consumer consumer) {
    this.consumer = consumer;
  }
 
  @Override
  public void run() {
    for(int i=0;i<10;i++){
      System.out.println(Thread.currentThread().getName()+" After Consume left:"+consumer.consume());
    }
 
  }
}
public class ConsumeClientA implements Consumer {
 
  private static int leftNum = 30;
 
  @Override
  public int consume() {
    int orgLeftNum = leftNum;
    Random random = new Random(System.currentTimeMillis());
    try {
      Thread.sleep(random.nextInt(3));
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
    orgLeftNum = orgLeftNum -1;
    leftNum = orgLeftNum;
    return leftNum;
  }
 
  public static void main(String[] args){
    Consumer consumer = new ConsumeClientA();
    Thread thread1 = new Thread(new ComsumeThread(consumer));
    Thread thread2 = new Thread(new ComsumeThread(consumer));
    Thread thread3 = new Thread(new ComsumeThread(consumer));
 
    thread1.start();
    thread2.start();
    thread3.start();
  }
}

ConsumeClientA是在沒有做任何線程安全處理,結(jié)果如下:

?
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
Thread-2 After Consume left:29
Thread-1 After Consume left:29
Thread-3 After Consume left:29
Thread-2 After Consume left:28
Thread-1 After Consume left:28
Thread-3 After Consume left:28
Thread-2 After Consume left:27
Thread-1 After Consume left:27
Thread-2 After Consume left:26
Thread-3 After Consume left:27
Thread-1 After Consume left:25
Thread-2 After Consume left:25
Thread-3 After Consume left:25
Thread-1 After Consume left:24
Thread-2 After Consume left:24
Thread-3 After Consume left:24
Thread-1 After Consume left:23
Thread-2 After Consume left:23
Thread-3 After Consume left:23
Thread-1 After Consume left:22
Thread-2 After Consume left:22
Thread-3 After Consume left:22
Thread-1 After Consume left:21
Thread-2 After Consume left:21
Thread-3 After Consume left:21
Thread-1 After Consume left:20
Thread-2 After Consume left:20
Thread-3 After Consume left:20
Thread-1 After Consume left:19
Thread-3 After Consume left:18

增加threadlocal處理,每個線程相互獨立,實現(xiàn)如下:

?
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
public class ConsumeClientB implements Consumer {
  private ThreadLocal<Integer> leftNumThreadLocal = new ThreadLocal<Integer>(){
    @Override
    protected Integer initialValue() {
      return 30;
    }
  };
 
  @Override
  public int consume() {
    int orgLeftNum = leftNumThreadLocal.get();
    Random random = new Random(System.currentTimeMillis());
    try {
      Thread.sleep(random.nextInt(3));
    } catch (InterruptedException e) {
      e.printStackTrace();
    }
    orgLeftNum = orgLeftNum -1;
    leftNumThreadLocal.set(orgLeftNum);
    return leftNumThreadLocal.get();
  }
 
  public static void main(String[] args){
    Consumer consumer = new ConsumeClientB();
    Thread thread1 = new Thread(new ComsumeThread(consumer));
    Thread thread2 = new Thread(new ComsumeThread(consumer));
    Thread thread3 = new Thread(new ComsumeThread(consumer));
 
    thread1.start();
    thread2.start();
    thread3.start();
  }
}

運行的結(jié)果如下:

?
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
Thread-1 After Consume left:29
Thread-3 After Consume left:29
Thread-2 After Consume left:29
Thread-1 After Consume left:28
Thread-3 After Consume left:28
Thread-2 After Consume left:28
Thread-1 After Consume left:27
Thread-3 After Consume left:27
Thread-2 After Consume left:27
Thread-1 After Consume left:26
Thread-3 After Consume left:26
Thread-2 After Consume left:26
Thread-1 After Consume left:25
Thread-3 After Consume left:25
Thread-2 After Consume left:25
Thread-1 After Consume left:24
Thread-3 After Consume left:24
Thread-2 After Consume left:24
Thread-1 After Consume left:23
Thread-3 After Consume left:23
Thread-2 After Consume left:23
Thread-1 After Consume left:22
Thread-3 After Consume left:22
Thread-2 After Consume left:22
Thread-1 After Consume left:21
Thread-3 After Consume left:21
Thread-2 After Consume left:21
Thread-1 After Consume left:20
Thread-3 After Consume left:20
Thread-2 After Consume left:20

每個線程擁有自己的獨立變量,相互隔離實現(xiàn)線程安全。

那ThreadLocal是怎樣實現(xiàn)這種線程隔離的線程安全的呢?

從ThreadLocal源碼可以看到,真正實現(xiàn)線程隔離,與線程掛鉤的,其實是ThreadLocal.ThreadLocalMap這個實現(xiàn)類,最明顯的體現(xiàn)就在于Thread類源碼的這樣一個變量申明說明了ThreadLocal.ThreadLocalMap與Thread的關(guān)系:

?
1
ThreadLocal.ThreadLocalMap threadLocals, inheritableThreadLocals;

Thread類是包含threadLocals對象的,ThreadLocal的具體實現(xiàn)就是根據(jù)提供的get,set等接口,對當前thread的threadLocals變量進行相關(guān)操作的,如get操作代碼如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public T get() {
  Thread t = Thread.currentThread();
  ThreadLocalMap map = getMap(t);
  if (map != null) {
    ThreadLocalMap.Entry e = map.getEntry(this);
    if (e != null)
      return (T)e.value;
  }
  return setInitialValue();
}
 
ThreadLocal.ThreadLocalMap getMap(Thread t) {
  return t.threadLocals;
}

可以看到,getMap()方法就是從當前thread獲取對應(yīng)的threadLocals變量,然后從這個ThreadLocal.ThreadLocalMap類型的threadLocals變量中獲取對應(yīng)線程中該ThreadLocal對象對應(yīng)的變量值。

set方法的操作也是一樣:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
public void set(T value) {
  Thread t = Thread.currentThread();
  ThreadLocal.ThreadLocalMap map = getMap(t);
  if(map != null) {
    map.set(this, value);
  } else {
    this.createMap(t, value);
  }
 
}
 
void createMap(Thread t, T firstValue) {
  t.threadLocals = new ThreadLocalMap(this, firstValue);
}
?
1
2
3
4
5
6
7
8
static class Entry extends WeakReference<ThreadLocal> {
      Object value;
 
      Entry(ThreadLocal var1, Object var2) {
        super(var1);
        this.value = var2;
      }
    }

ThreadLocalMap中存的是內(nèi)部類Entry的數(shù)組,Entry是繼承WeakReference實現(xiàn),WeakReference的好處是保存對象引用,而又不干擾該對象被GC回收,線程執(zhí)行完回收threadLocals變量時不會受到Entry封裝的變量的干擾。

而且ThreadLocalMap中的key是ThreadLocal,所以一個ThreadLocal對象只能在一個Thread對象中保存一個ThreadLocal的value。

綜上,很多人說ThreadLocal的實現(xiàn)是ThreadLocalMap中存Thread對象為key,變量為value的map結(jié)構(gòu),其實是錯誤的。

感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!

原文鏈接:http://blog.csdn.net/jamesjxin/article/details/45173201

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 四虎音影 | 四虎影院在线免费观看 | 日韩福利一区 | 欧美一级鲁丝片免费看 | 国产福利微拍精品一区二区 | bdsm中国精品调教 | 无码爽死成人777在线观看网站 | 国产精品嫩草影院在线看 | japanesepooping脱粪 | 精品手机在线1卡二卡3卡四卡 | 黄a 大片a v 永久免费 | 久久伊人精品青青草原2021 | 精油按摩日本 | 师尊被各种play打屁股 | 成人女人天堂午夜视频 | 国四虎影永久 | 69pao强力打造免费高速 | 闺蜜的样子小说安沁在线阅读 | 91精品国产91久久久久 | 国产人成77777视频网站 | 欧美日韩国产另类一区二区三区 | 激情视频激情小说 | pron在线观看| 国产成人盗摄精品 | 91制片厂制作传媒网站破解 | 亚洲天堂日韩在线 | 亚洲日韩精品欧美一区二区 | 99国产精品久久久久久久... | 国产v日韩v欧美v精品专区 | 久久青草免费91线频观看站街 | 日韩在线免费播放 | 黑人粗长大战亚洲女 | 天天干夜夜玩 | 日韩欧美国产成人 | 精品性影院一区二区三区内射 | 国内精品久久久久久中文字幕 | 国产91精选学生在线观看 | videos护士有奶水 | 奇米影视奇米色777欧美 | 深夜在线观看网站 | 亚洲精品第一国产综合 |