`
java-mans
  • 浏览: 11431467 次
文章分类
社区版块
存档分类
最新评论

杂谈ThreadLocal

 
阅读更多

ThreadLocal很容易让人认为是一个"本地线程"。其实ThreadLocal并不是一个Thread,而是Thread的局部变量。ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。

public class ThreadLocalDemo implementsRunnable {
// 创建线程局部变量studentLocal,在后面你会发现用来保存Student对象
privatestatic final ThreadLocal studentLocal = new ThreadLocal();
publicstatic void main(String[] agrs) {
ThreadLocalDemo td =new ThreadLocalDemo();
Thread t1 =new Thread(td, "a");
Thread t2 =new Thread(td, "b");
t1.start();
t2.start();
}
publicvoid run() {
accessStudent();
}
/**
* 示例业务方法,用来测试
*/
publicvoid accessStudent() {
// 获取当前线程的名字
String currentThreadName = Thread.currentThread().getName();
System.out.println(currentThreadName +" is running!");
// 产生一个随机数并打印
Random random =new Random();
intage = random.nextInt(100);
System.out.println("thread "+ currentThreadName + " set age to:"+ age);
// 获取一个Student对象,并将随机数年龄插入到对象属性中
Person student = getStudent();
student.setAge(age);
System.out.println("thread "+ currentThreadName + " first read age is:"+ student.getAge());
try{
Thread.sleep(500);
}catch (InterruptedException ex) {
ex.printStackTrace();
}
System.out.println("thread "+ currentThreadName + " second read age is:"+ student.getAge());
}
protectedPerson getStudent() {
// 获取本地线程变量并强制转换为Student类型
Person student = (Person) studentLocal.get();
// 线程首次执行此方法的时候,studentLocal.get()肯定为null
if(student == null) {
// 创建一个Student对象,并保存到本地线程变量studentLocal中
student =new Person();
studentLocal.set(student);
}
returnstudent;
}
}

ThreadLocal和线程同步机制的区别:

同步机制中,通过对象的锁机制保证同一时间只有一个线程访问变量。这时该变量是多个线程共享的,使用同步机制要求程序慎密地分析什么时候对变量进行读写,什么时候需要锁定某个对象,什么时候释放对象锁等繁杂的问题,程序设计和编写难度相对较大。

而ThreadLocal则从另一个角度来解决多线程的并发访问。ThreadLocal会为每一个线程提供一个独立的变量副本,从而隔离了多个线程对数据的访问冲突。因为每一个线程都拥有自己的变量副本,从而也就没有必要对该变量进行同步了。ThreadLocal提供了线程安全的共享对象,在编写多线程代码时,可以把不安全的变量封装进ThreadLocal。

总结:

ThreadLocal使用场合主要解决多线程中数据数据因并发产生不一致问题。ThreadLocal为每个线程的中并发访问的数据提供一个副本,通过访问副本来运行业务,这样的结果是耗费了内存,单大大减少了线程同步所带来性能消耗,也减少了线程并发控制的复杂度。

ThreadLocal不能使用原子类型,只能使用Object类型。ThreadLocal的使用比synchronized要简单得多。

ThreadLocal和Synchonized都用于解决多线程并发访问。但是ThreadLocal与synchronized有本质的区别。synchronized是利用锁的机制,使变量或代码块在某一时该只能被一个线程访问。而ThreadLocal为每一个线程都提供了变量的副本,使得每个线程在某一时间访问到的并不是同一个对象,这样就隔离了多个线程对数据的数据共享。而Synchronized却正好相反,它用于在多个线程间通信时能够获得数据共享。

Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics