標籤:star boolean 線程 read 工作過程 public main instance 分析
ι 著作權聲明:本文為博主原創文章,未經博主允許不得轉載。
在上一篇文章中,我們已經提到了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=true06-17 16:46:04.663 5300-5317/com.bella.androidart E/ThreadLocalActivity: [Thread1]mBooleanThreadLocal=false06-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的工作過程