static final ThreadLocal
変数はスレッドセーフです。
static
ThreadLocal変数を、それぞれのスレッドに対してのみ複数のクラスで利用できるようにします。これは、複数のクラスにまたがるそれぞれのスレッドローカル変数の一種のグローバル変数の宣言です。
このスレッドの安全性は、次のコードサンプルで確認できます。
CurrentUser
-現在のユーザーIDをThreadLocalに保存します
TestService
-メソッドを使用した単純なサービスgetUser()
-CurrentUserから現在のユーザーを取得します。
TestThread
-このクラスは、複数のスレッドを作成し、ユーザーIDを同時に設定するために使用されます
。
public class CurrentUser
public class CurrentUser {
private static final ThreadLocal<String> CURRENT = new ThreadLocal<String>();
public static ThreadLocal<String> getCurrent() {
return CURRENT;
}
public static void setCurrent(String user) {
CURRENT.set(user);
}
}
public class TestService {
public String getUser() {
return CurrentUser.getCurrent().get();
}
}
。
import java.util.ArrayList;
import java.util.List;
public class TestThread {
public static void main(String[] args) {
List<Integer> integerList = new ArrayList<>();
//creates a List of 100 integers
for (int i = 0; i < 100; i++) {
integerList.add(i);
}
//parallel stream to test concurrent thread execution
integerList.parallelStream().forEach(intValue -> {
//All concurrent thread will set the user as "intValue"
CurrentUser.setCurrent("" + intValue);
//Thread creates a sample instance for TestService class
TestService testService = new TestService();
//Print the respective thread name along with "intValue" value and current user.
System.out.println("Start-"+Thread.currentThread().getName()+"->"+intValue + "->" + testService.getUser());
try {
//all concurrent thread will wait for 3 seconds
Thread.sleep(3000l);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//Print the respective thread name along with "intValue" value and current user.
System.out.println("End-"+Thread.currentThread().getName()+"->"+intValue + "->" + testService.getUser());
});
}
}
。
TestThreadメインクラスを実行します。出力-
Start-main->62->62
Start-ForkJoinPool.commonPool-worker-2->31->31
Start-ForkJoinPool.commonPool-worker-3->81->81
Start-ForkJoinPool.commonPool-worker-1->87->87
End-main->62->62
End-ForkJoinPool.commonPool-worker-1->87->87
End-ForkJoinPool.commonPool-worker-2->31->31
End-ForkJoinPool.commonPool-worker-3->81->81
Start-ForkJoinPool.commonPool-worker-2->32->32
Start-ForkJoinPool.commonPool-worker-3->82->82
Start-ForkJoinPool.commonPool-worker-1->88->88
Start-main->63->63
End-ForkJoinPool.commonPool-worker-1->88->88
End-main->63->63
...
分析のまとめ
- 「メイン」スレッドが開始し、現在のユーザーを「62」に設定します。並列に「ForkJoinPool.commonPool-worker-2」スレッドが開始し、現在のユーザーを「31」に設定します。並列に「ForkJoinPool.commonPool-worker-3」スレッドが開始し、現在に設定します"81"としてユーザー、並列に "ForkJoinPool.commonPool-worker-1"スレッドが開始し、現在のユーザーを "87"として設定Start-main-> 62-> 62 Start-ForkJoinPool.commonPool-worker-2-> 31-> 31 Start-ForkJoinPool.commonPool-worker-3-> 81-> 81 Start-ForkJoinPool.commonPool-worker-1-> 87-> 87
- 上記のすべてのスレッドは3秒間スリープします
main
実行が終了して現在のユーザーが「62」として印刷されます。並列ForkJoinPool.commonPool-worker-1
実行が終了して現在のユーザーが「87」として印刷されます。並列ForkJoinPool.commonPool-worker-2
実行が終了して現在のユーザーが「31」として印刷されます。並列ForkJoinPool.commonPool-worker-3
実行が終了して現在のユーザーが「81」として印刷されます
推論
並行スレッドは、「static final ThreadLocal」として宣言されている場合でも、正しいユーザーIDを取得できます。