我试图找出如何在Java 8并行流中复制ThreadLocal值。
所以如果我们考虑这个:
public class ThreadLocalTest {
public static void main(String[] args) {
ThreadContext.set("MAIN");
System.out.printf("Main Thread: %s\n", ThreadContext.get());
IntStream.range(0,8).boxed().parallel().forEach(n -> {
System.out.printf("Parallel Consumer - %d: %s\n", n, ThreadContext.get());
});
}
private static class ThreadContext {
private static ThreadLocal<String> val = ThreadLocal.withInitial(() -> "empty");
public ThreadContext() {
}
public static String get() {
return val.get();
}
public static void set(String x) {
ThreadContext.val.set(x);
}
}
}
哪个输出
Main Thread: MAIN
Parallel Consumer - 5: MAIN
Parallel Consumer - 4: MAIN
Parallel Consumer - 7: empty
Parallel Consumer - 3: empty
Parallel Consumer - 1: empty
Parallel Consumer - 6: empty
Parallel Consumer - 2: empty
Parallel Consumer - 0: MAIN
有没有办法让我将main()方法中的ThreadLocal克隆到为每个并行执行生成的线程中?
这样我的结果就是:
Main Thread: MAIN
Parallel Consumer - 5: MAIN
Parallel Consumer - 4: MAIN
Parallel Consumer - 7: MAIN
Parallel Consumer - 3: MAIN
Parallel Consumer - 1: MAIN
Parallel Consumer - 6: MAIN
Parallel Consumer - 2: MAIN
Parallel Consumer - 0: MAIN
而不是第一个?
答案 0 :(得分:10)
As Louis has stated in the comments,你的例子很可能被简化为在lambda表达式中捕获局部变量的值
public static void main(String[] args) {
String value = "MAIN";
System.out.printf("Main Thread: %s\n", value);
IntStream.range(0,8).boxed().parallel().forEach(n -> {
System.out.printf("Parallel Consumer - %d: %s\n", n, value);
});
}
从您的示例中可以看出完整的用例是什么。
如果您确切知道主线程中哪些线程已启动,您可以考虑使用InheritableThreadLocal
此类扩展
ThreadLocal
以提供值的继承 父线程到子线程:当创建子线程时, child接收所有可继承的thread-local的初始值 父对象具有值的变量。
在您的情况下,将val
声明为InheritableThreadLocal
,因为在Thread
内为parallel()
创建的ForkJoinPool#commonPool()
个实例是懒散创建的,所以他们都会继承自set
方法(和线程)中的值main
。
如果您在原始线程中设置commonPool
值之前以某种方式使用parallel
(或调用InhertiableThreadLocal
终端操作的任何池),则不会出现这种情况