I want to fly higher
          programming Explorer
          posts - 114,comments - 263,trackbacks - 0
          1.代碼示例
          package com.landon.mavs.example.concurrent;

          import java.util.concurrent.ThreadLocalRandom;

          /**
           * 
           * ThreadLocal例子
           * 
           * <pre>
           *     1.線程局部變量.其完全不提供鎖.以空間換時間的手段,為每個線程提供變量的獨立副本,以保證線程的安全.
           *     2.接口:
           *         public T get() 返回此線程局部變量的當前線程副本中的值
           *         public void set(T value) 將此線程局部變量的當前線程副本中的值設置為指定值
           *         public void remove() 移除此線程局部變量當前線程的值
           *         protected T initialValue() 返回此線程局部變量的當前線程的“初始值”
           *  3.源碼:
           *       // 1.取得當前調用線程t 2.根據t獲得ThreadLocalMap【Entry(ThreadLocal k, Object v)】(即每個線程都有一個threadLocals)
           *       // 3. 如果map為不null,則將this和value設置到map;否則創建map
           *       public void set(T value) {
           *         Thread t = Thread.currentThread();
           *         ThreadLocalMap map = getMap(t);
           *         if (map != null)
           *             map.set(this, value);
           *         else
           *             createMap(t, value);
           *     }
           *     
           *     // 1.取得當前調用線程t 2.取得t線程關聯的ThreadLocalMap
           *     // 3.如果map不為null則將this作為key得到Entry.然后得到值.
           *     // 4.如果map為null則返回initialValue(默認為null,同時會將initialValue set至當前線程)
           *     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();
           *     }.
           *  4.注意:
           *      1.不同線程間的對象副本并不是由ThreadLocal創建的.因為set方法未生成任何value的副本.所以將一個對象的實例設置到不同線程的ThreadLocal中,同樣無法保證線程安全.
           *  所以需要每個線程內創建副本,然后設置到ThreadLocal中(即ThreadLocal只是一個key而已.).
           *      2.從get方法的最后一句看出,如果沒有set,則第一次調用get方法時會調用initialValue;可使用匿名內部類覆寫此方法,如果不希望初始值為null.另外如果調用了remove方法,即從
           *  ThreadLocalMap移除ThreadLocal key,則再次調用get方法時依然會調用initialValue.
           *      3.ThreadLocal實例通常是類中的 private static final字段(1.其只是用來做key,和當前線程綁定,所以用static方便 2.通常用靜態方法直接獲取局部變量實例,
           *  如ThreadLocalRandom),它們希望將狀態與某一個線程相關聯.
           *  
           *  5.總結:
           *      簡單來說,ThreadLocal就是一個當前線程局部變量表的一個key,該key對應的V是new出來的一個新值,每個線程的都不一樣.這樣
           *  當前線程拿到ThreadLocal.get->則直接獲得線程當前局部變量表對應的V.(第一次調用get時會將會將initialValue set,
           *  所以初始化ThreadLocal的時候可覆寫initialValue).
           * </pre>
           * 
           * @author landon
           * 
           
          */

          public class ThreadLocalExample {
              
          // 一個線程局部變量
              private static final ThreadLocal<ThreadLocalVar> localVar = new ThreadLocal<>();

              
          // 線程局部變量內部的V
              private static class ThreadLocalVar {
                  
          public int var;
              }


              
          // 初始化的時候指定了localVar2的變量值
              private static final ThreadLocal<ThreadLocalVar> localVar2 = new ThreadLocal<ThreadLocalVar>() {
                  
          protected ThreadLocalVar initialValue() {
                      
          return new ThreadLocalVar();
                  }

              }
          ;

              
          // 一個任務
              private static class ThreadLocalTask implements Runnable {
                  @Override
                  
          public void run() {
                      
          // 注意這里必須要new一個值,然后set.->即set了當前調用線程中局部變量的值.
                      ThreadLocalVar var = new ThreadLocalVar();
                      localVar.set(var);

                      
          // 隨機一個局部變量的值.這里用到了ThreadLocalRandom,一個活生生的例子

                      
          // public static ThreadLocalRandom current() {
                      
          // return localRandom.get();
                      
          // }

                      
          // 這里初始化了一個線程局部變量值,是new了一個ThreadLocalRandom.這個很關鍵.
                      
          // private static final ThreadLocal<ThreadLocalRandom> localRandom =
                      
          // new ThreadLocal<ThreadLocalRandom>() {
                      
          // protected ThreadLocalRandom initialValue() {
                      
          // return new ThreadLocalRandom();
                      
          // }
                      
          // };
                      var.var = ThreadLocalRandom.current().nextInt();

                      
          // 打印當前調用線程,即當前調用線程的線程局部變量的值.
                      System.out.println("curThread:" + Thread.currentThread().getName()
                              
          + " localVar:" + localVar.get().var);

                      
          // 操作localVar2,將當前線程的id復制給localvar2
                      localVar2.get().var = (int) Thread.currentThread().getId();

                      System.out.println("curThread:" + Thread.currentThread().getName()
                              
          + " localVar2:" + localVar2.get().var);
                  }

              }


              
          // 從輸出看,ThreadLocalExample#localVar如果不是線程局部變量則多線程操作時,一定要加鎖的.但是如果是線程局部變量,則其只用來做一個key而已,局部變量的值是new出來
              
          // 并設置,所以每個線程的ThreadLocal的get值都是互相沒有任何關系的。
              
          // 多線程之間可隨意操作localVar.
              
          // 同理localvar2
              public static void main(String[] args) {
                  
          for (int i = 0; i < 5; i++{
                      
          new Thread(new ThreadLocalTask(), "Thread-" + i).start();
                  }


                  
          // 主線程操作
                  localVar.set(new ThreadLocalVar());
                  localVar.get().var = 100;

                  System.out.println("curThread:" + Thread.currentThread().getName()
                          
          + " localVar:" + localVar.get().var);

                  
          // 主線程操作localvar2
                  localVar2.get().var = 201;

                  System.out.println(
          "curThread:" + Thread.currentThread().getName()
                          
          + " localVar:" + localVar2.get().var);
              }

          }



          2.本篇主要介紹了ThreadLocal的用法,作為解決線程安全的一種方法.ThreadLocalRandom就是一個典型例子.

          posted on 2014-03-05 15:17 landon 閱讀(1829) 評論(0)  編輯  收藏 所屬分類: Program
          主站蜘蛛池模板: 乐清市| 普宁市| 阳西县| 榆林市| 拉萨市| 崇文区| 金塔县| 长汀县| 汪清县| 元阳县| 岗巴县| 墨江| 兴安县| 凉山| 仪征市| 广安市| 大埔区| 阿瓦提县| 汽车| 阿勒泰市| 唐山市| 镇沅| 西和县| 庆阳市| 博兴县| 修水县| 微博| 依兰县| 天气| 乐昌市| 开鲁县| 乌苏市| 乐至县| 五台县| 雷州市| 高尔夫| 莒南县| 麻江县| 静乐县| 霞浦县| 长沙市|