ThreadLocal创建的多个副本是同一个对象吗?
不是同一个对象。ThreadLocal解决线程安全问题的原理是为每一个线程建立一份资源的副本,保证资源的独享性,那么这些副本是同一个对象吗,当然不是,如果是的话,为什么各个线程对副本进行修改后不会相互影响呢。验证如下。
测试代码:
public class UseThreadLocal {
public static ThreadLocal<Object> threadLocal = new ThreadLocal<Object>(){
@Override
protected Object initialValue() {
return new Object();
}
};
/**
* 运行三个线程
*/
public void StartThreadArray(){
Thread[] runs = new Thread[3];
for (int i = 0; i < runs.length; i++) {
runs[i] = new Thread(new TestThreadLocal(i));
}
for (int i = 0; i < runs.length; i++) {
runs[i].start();
}
}
public static class TestThreadLocal implements Runnable{
int id;
public TestThreadLocal(int id){
this.id = id;
}
@Override
public void run() {
System.out.println(Thread.currentThread().getName()+":start");
Object str = threadLocal.get();
// threadLocal.set(str+id);
// System.out.println(Thread.currentThread().getName()+" "+str.getBytes());
System.out.println(Thread.currentThread().getName()+" "+str.hashCode());
System.out.println(Thread.currentThread().getName()+" "+threadLocal.get());
}
}
public static void main(String[] args) {
UseThreadLocal test = new UseThreadLocal();
test.StartThreadArray();
}
}
测试结果:
Thread-1:start
Thread-2:start
Thread-0:start
Thread-2 535693062
Thread-1 1708263019
Thread-0 105270561
Thread-2 java.lang.Object@1fee0706
Thread-1 java.lang.Object@65d2066b
Thread-0 java.lang.Object@6464d21
从测试结果可以看出,ThreadLocal在创建副本时是在堆内存里创建了新的对象。
tips: 若ThreadLocal使用String类型验证,则不能用hashcode来比较,因为String类中hashcode方法重写过,可用getBytes()方法来看是不是同一个对象。
本文解析ThreadLocal如何为每个线程创建独立副本,确保资源隔离。通过测试实例说明,不同线程对副本操作互不影响。关键在于新对象的分配,而非共享对象。

445

被折叠的 条评论
为什么被折叠?



