1. 程式人生 > >【原創】源碼角度分析Android的消息機制系列(二)——ThreadLocal的工作過程

【原創】源碼角度分析Android的消息機制系列(二)——ThreadLocal的工作過程

機制 simple hand 這就是 數據存儲 read etc lena 並且

ι 版權聲明:本文為博主原創文章,未經博主允許不得轉載。

在上一篇文章中,我們已經提到了ThreadLocal,它並非線程,而是在線程中存儲數據用的。數據存儲以後,只能在指定的線程中獲取到數據,對於其他線程來說是無法獲取到數據的,也就是說ThreadLocal可以在多個線程中互不幹擾地存儲和修改數據。基於ThreadLocal的這一特點,那麽當我們在開發中,需要將某些數據以線程作為作用域,並且不同線程具有不同的數據副本的時候,就可以考慮采用ThreadLocal了。

Android的消息機制中也用到了ThreadLocal。Handler需要獲取到當前線程的Looper,但是不同線程中的Looper是不同的,此時就可以通過ThreadLocal來實現Looper在線程中的存取了。

下面通過例子來觀察下ThreadLocal存儲數據的特點。

首先定義一個ThreadLocal對象,然後分別在主線程,子線程1和子線程2中設置和訪問它的值,代碼如下所示:

    private final String TAG=getClass().getSimpleName(); //獲取當前類名
    private ThreadLocal<Boolean> mBooleanThreadLocal=new ThreadLocal<>();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        
super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mBooleanThreadLocal.set(true); Log.e(TAG, "[mainThread]mBooleanThreadLocal="+mBooleanThreadLocal.get() ); new Thread("Thread1"){ @Override public void run() {
super.run(); mBooleanThreadLocal.set(false); Log.e(TAG, "[Thread1]mBooleanThreadLocal="+mBooleanThreadLocal.get() ); } }.start(); new Thread("Thread2"){ @Override public void run() { super.run(); Log.e(TAG, "[Thread2]mBooleanThreadLocal="+mBooleanThreadLocal.get() ); } }.start(); }

日誌如下所示:

06-17 16:46:04.663 5300-5300/com.bella.androidart E/ThreadLocalActivity: [mainThread]mBooleanThreadLocal=true
06-17 16:46:04.663 5300-5317/com.bella.androidart E/ThreadLocalActivity: [Thread1]mBooleanThreadLocal=false
06-17 16:46:04.679 5300-5318/com.bella.androidart E/ThreadLocalActivity: [Thread2]mBooleanThreadLocal=null

從上面日誌可以看出,雖然在不同線程中訪問的是同一個ThreadLocal對象,但是它們通過ThreadLocal獲取到的值卻是不一樣的,這就是ThreadLocal的奇妙之處。ThreadLocal之所以有這麽奇妙的效果,是因為不同線程訪問同一個ThreadLocal的get方法,ThreadLocal內部會從各自的線程中取出一個數組,然後再從數組中根據當前ThreadLocal的索引去查找出對應的value值。很顯然,不同線程中的數組是不同的。

【原創】源碼角度分析Android的消息機制系列(二)——ThreadLocal的工作過程