<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    posts - 1,  comments - 25,  trackbacks - 0

        ThreadLocal的核心思想很簡單:為每個獨立的線程提供一個變量的副本。

        ThreadLocal則使用了“拷貝副本”的方式,人人有份,你用你的,我用我的,大家互不影響,是“以空間換時間”。每個線程修改變量時,實際上修改的是變量的副本,不怕影響到其它線程。

        為了加深對ThreadLocal的理解,下面我使用一個例子來演示ThreadLocal如何隔離線程間的變量訪問和修改:

    【1】SerialNum類

    package example.thread.threadLocal;

    public class SerialNum {

        private static int nextSerialNum = 1;

        @SuppressWarnings("unchecked")
        private static ThreadLocal serialNum = new ThreadLocal() {
            protected synchronized Object initialValue() {
                return new Integer(nextSerialNum++);      
            }                                                           
        };

        public static int get() {
            return ((Integer) (serialNum.get())).intValue();
        }
        
        @SuppressWarnings("unchecked")
        public static void set(Integer newSerial){
            serialNum.set(newSerial);
        }
    }

    【2】GetSerialNumThread
    package example.thread.threadLocal;

    public class GetSerialNumThread implements Runnable {

        public static void main(String args[]) {

            GetSerialNumThread serialNumGetter = new GetSerialNumThread();
            Thread t1 = new Thread(serialNumGetter, "Thread A");
            Thread t2 = new Thread(serialNumGetter, "Thread B");
            t1.start();
            try {
                t1.join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }    
            t2.start();            
        }

        public void run() {
            int mySerialNum = getSerialNum();
            System.out.println("線程 " + Thread.currentThread().getName()
                    + " 獲取到的序列號是" + mySerialNum);
            System.out.println("線程 " + Thread.currentThread().getName()
                    + " 修改了序列號為" + (mySerialNum * 3));
            setSerialNum(mySerialNum * 3);
            System.out.println("線程 " + Thread.currentThread().getName()
                    + " 再次獲得的序列號是" + getSerialNum());
        }

        private int getSerialNum() {
            return SerialNum.get();
        }

        private void setSerialNum(int newSerialNum) {
            SerialNum.set(new Integer(newSerialNum));
        }
    }

    運行的結果如下:
    線程 Thread A 獲取到的序列號是1
    線程 Thread A 修改了序列號為3
    線程 Thread A 再次獲得的序列號是3
    線程 Thread B 獲取到的序列號是2
    線程 Thread B 修改了序列號為6
    線程 Thread B 再次獲得的序列號是6

        可見第一個線程在調用SerialNum.set(int)方法修改static變量時,其實修改的是它自己的副本,而不是修改本地變量,第二個線程在初始化的時候拿到的序列號是2而不是7。

        為什么會這樣呢?明明serialNum是靜態變量啊?其實我們只需要看看ThreadLocal的內部構造就知道了:

    A. ThreadLocal的get()方法:
     /**
         * Returns the value in the current thread's copy of this thread-local
         * variable.  Creates and initializes the copy if this is the first time
         * the thread has called this method.
         *
         * @return the current thread's value of this thread-local
         */
        public T get() {
            Thread t = Thread.currentThread();
            ThreadLocalMap map = getMap(t);
            if (map != null)
                return (T)map.get(this);

            // Maps are constructed lazily.  if the map for this thread
            // doesn't exist, create it, with this ThreadLocal and its
            // initial value as its only entry.
            T value = initialValue();
            createMap(t, value);
            return value;
        }

    B. ThreadLocal的set()方法:
    /**
         * Sets the current thread's copy of this thread-local variable
         * to the specified value.  Many applications will have no need for
         * this functionality, relying solely on the {@link #initialValue}
         * method to set the values of thread-locals.
         *
         * @param value the value to be stored in the current threads' copy of
         *        this thread-local.
         */
        public void set(T value) {
            Thread t = Thread.currentThread();
            ThreadLocalMap map = getMap(t);
            if (map != null)
                map.set(this, value);
            else
                createMap(t, value);
        }

        可以看到ThreadLocal在內部維護了一個Map,將變量的值和線程綁定起來,get/set方法都是對該線程對應的value進行操作,所以不會影響到其它線程。
    posted on 2010-12-09 19:27 Daniel 閱讀(204) 評論(0)  編輯  收藏 所屬分類: CoreJava
    <2025年5月>
    27282930123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    常用鏈接

    留言簿(3)

    隨筆檔案

    文章分類

    文章檔案

    相冊

    搜索

    •  

    最新評論

    主站蜘蛛池模板: 久久伊人亚洲AV无码网站| 黄色三级三级免费看| 国产亚洲色视频在线| 中文字幕无码不卡免费视频| 国产午夜无码片免费| 亚洲精品无码久久久久久| 亚洲小视频在线观看| 国产一区二区三区在线免费| 精品熟女少妇av免费久久| 国产综合激情在线亚洲第一页| 久久亚洲国产伦理| 亚洲国产成人久久综合野外| 好吊妞在线成人免费| 在线看片韩国免费人成视频| 99久久国产精品免费一区二区| 亚洲国产精品ⅴa在线观看| 亚洲国产精品久久丫| 久久亚洲精品无码AV红樱桃| 亚洲爆乳无码专区| 国产偷窥女洗浴在线观看亚洲| 国产午夜无码视频免费网站| 日韩一级在线播放免费观看| 老司机在线免费视频| 99精品国产免费久久久久久下载 | 中文成人久久久久影院免费观看| 亚洲国产成人无码AV在线影院| 亚洲一区二区三区深夜天堂 | a级毛片免费播放| 国产JIZZ中国JIZZ免费看| 深夜特黄a级毛片免费播放| 国产综合激情在线亚洲第一页| 永久免费毛片在线播放| 青青草97国产精品免费观看| 亚洲av成本人无码网站| 成人亚洲国产va天堂| 亚洲国产综合精品中文第一| 久久久久久亚洲精品影院| 亚洲欧洲尹人香蕉综合| 亚洲国产综合精品| 亚洲国产精品综合久久20| 亚洲13又紧又嫩又水多|