<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)

    隨筆檔案

    文章分類

    文章檔案

    相冊

    搜索

    •  

    最新評論

    主站蜘蛛池模板: 亚洲色大成网站www永久一区| 亚洲av无码专区在线播放| 免费A级毛片av无码| 色婷婷亚洲一区二区三区| 亚洲视频一区在线观看| 亚洲人成网站色在线入口| 日韩精品福利片午夜免费观着| 亚洲日韩乱码久久久久久| 三级毛片在线免费观看| 一级一看免费完整版毛片| 亚洲成AV人片一区二区密柚| 免费无码专区毛片高潮喷水| 亚洲一区二区三区深夜天堂| 亚洲国产精品成人精品无码区| 亚洲国产成人影院播放| 暖暖免费高清日本中文| 7723日本高清完整版免费| 久久久久国产精品免费网站| 亚洲精品黄色视频在线观看免费资源| 亚洲人片在线观看天堂无码| 亚洲国产精品yw在线观看| 18亚洲男同志videos网站| 亚洲AV午夜成人影院老师机影院 | 国产精品国产亚洲精品看不卡| 免费a级毛片无码a∨性按摩| 中文字幕在线视频免费观看| 午夜免费国产体验区免费的| 成人精品国产亚洲欧洲| 色窝窝亚洲av网| 国产精品成人亚洲| 久久亚洲精品中文字幕三区| 国产亚洲一区区二区在线| 四虎最新永久免费视频| 99re在线视频免费观看| 99久在线国内在线播放免费观看 | 亚洲AV永久无码精品| 亚洲中文字幕在线乱码| 亚洲精品国产精品乱码不99| 亚洲精品V欧洲精品V日韩精品| 中文亚洲AV片在线观看不卡| 亚洲线精品一区二区三区|