標籤:
共用資料是並發程式最核心的問題之一。下面我們看一個Deom感受一下多線程對屬性的影響。
代碼:
UnsafeTask.java
package com.tgb.klx.thread;import java.text.SimpleDateFormat;import java.util.Date;import java.util.concurrent.TimeUnit;public class UnsafeTask implements Runnable {private String startDate;@Overridepublic void run() {SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");startDate=sdf.format( new Date());System.out.println("開始線程:"+Thread.currentThread().getId()+",開始的時間:"+startDate);try {TimeUnit.SECONDS.sleep((int)Math.rint(Math.random()*10));} catch (InterruptedException e) {e.printStackTrace();}System.out.println("結束線程:"+Thread.currentThread().getId()+",開始的時間:"+startDate);}}
Core.java
package com.tgb.klx.thread;import java.util.concurrent.TimeUnit;public class Core {public static void main(String[] args) {UnsafeTask task=new UnsafeTask();for(int i=0;i<10;i++){Thread thread=new Thread(task);thread.start();try {TimeUnit.SECONDS.sleep(2);} catch (InterruptedException e) {e.printStackTrace();}}}}
運行結果:
可以看到,最後一個線程開始的時間,是剩餘線程結束的時間,明顯的一個線程啟動的時間被其他線程修改了。
那麼,我們通過什麼方式能保證一個線程的局部變數不會被其他線程修改呢?下面我們使用ThreadLocal,來保護相線程的屬性不被其他線程修改:
代碼:
SafeTask.java
package com.tgb.klx.thread;import java.text.SimpleDateFormat;import java.util.Date;import java.util.concurrent.TimeUnit;public class SafeTask implements Runnable {private static ThreadLocal<String> startDate=new ThreadLocal<String>(){protected String initialValue(){SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");return sdf.format( new Date());}};@Overridepublic void run() {System.out.println("開始線程:"+Thread.currentThread().getId()+",開始的時間:"+startDate.get());try {TimeUnit.SECONDS.sleep((int)Math.rint(Math.random()*10));} catch (InterruptedException e) {e.printStackTrace();}System.out.println("結束線程:"+Thread.currentThread().getId()+",開始的時間:"+startDate.get());}}
運行結果:
我們保證了線程內的局部變數的安全性。ThreadLocal是JDK 1.2的版本中就提供java.lang.ThreadLocal,ThreadLocal為解決多線程程式的並發問題提供了一種新的思路。使用這個工具類可以很簡潔地編寫出優美的多線程程式,ThreadLocal並不是一個Thread,而是Thread的局部變數。
Java並發編程--線程局部變數使用