早在JDK 1.2
的版本中就提供java.lang.ThreadLocal
,ThreadLocal
为解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序。
ThreadLocal
很容易让人望文生义,想当然地认为是一个“
本地线程”
。其实,ThreadLocal
并不是一个Thread
,而是Thread
的局部变量,也许把它命名为ThreadLocalVariable
更容易让人理解一些。
当使用ThreadLocal
维护变量时,ThreadLocal
为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。
从线程的角度看,目标变量就象是线程的本地变量,这也是类名中“Local”
所要表达的意思。
线程局部变量并不是Java
的新发明,很多语言(如IBM IBM XL FORTRAN
)在语法层面就提供线程局部变量。在Java
中没有提供在语言级支持,而是变相地通过ThreadLocal
的类提供支持。
所以,在Java
中编写线程局部变量的代码相对来说要笨拙一些,因此造成线程局部变量没有在Java
开发者中得到很好的普及。
ThreadLocal
的接口方法
ThreadLocal
类接口很简单,只有4
个方法,我们先来了解一下:
设置当前线程的线程局部变量的值。
该方法返回当前线程所对应的线程局部变量。
将当前线程局部变量的值删除,目的是为了减少内存的占用,该方法是JDK 5.0
新增的方法。需要指出的是,当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法清除线程的局部变量并不是必须的操作,但它可以加快内存回收的速度。
-
protected Object
initialValue()
返回该线程局部变量的初始值,该方法是一个protected
的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1
次调用get()
或set(Object)
时才执行,并且仅执行1
次
。ThreadLocal
中的缺省实现直接返回一个null
。
值得一提的是,在JDK5.0
中,ThreadLocal
已经支持泛型,该类的类名已经变为ThreadLocal<T>
。API
方法 也相应进行了调整,新版本的API
方法分别是void set(T value)
、T get()
以及T initialValue()
。
ThreadLocal
是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单:在ThreadLocal
类中有一个Map
,用于存储每一个线程的变量副本,Map
中元素的键为线程对象,而值对应线程的变量副本。我们自己就可以提供一个简单的实现版本:
代码清单
1 SimpleThreadLocal
public class SimpleThreadLocal {
private Map valueMap = Collections.synchronizedMap(new HashMap());
public void set(Object newValue) {
valueMap.put(Thread.currentThread(), newValue
);①
键为线程对象,值为本线程的变量副本
}
public Object get() {
Thread currentThread = Thread.currentThread();
Object o = valueMap.get(currentThread);②
返回本线程对应的变量
if (o == null && !valueMap.containsKey(currentThread)) {③
如果在Map
中不存在,放到Map
中保存起来。
o = initialValue();
valueMap.put(currentThread, o);
}
return o;
}
public void remove() {
valueMap.remove(Thread.currentThread());
}
public Object initialValue() {
return null;
}
}
虽然代码清单9‑3
这个ThreadLocal
实现版本显得比较幼稚,但它和JDK
所提供的ThreadLocal
类在实现思路上是相近的。
一个TheadLocal
实例
下面,我们通过一个具体的实例了解一下ThreadLocal
的具体使用方法。
代码清单
2 SequenceNumber
package com.baobaotao.basic;
public class SequenceNumber {
①
通过匿名内部类覆盖ThreadLocal
的initialValue()
方法,指定初始值
private static ThreadLocal<Integer> seqNum = new ThreadLocal<Integer>
(){
public Integer initialValue(){
return 0;
}
};
②
获取下一个序列值
public int getNextNum(){
seqNum.set(seqNum.get()+1);
return seqNum.get();
}
public static void main(String[] args)
{
SequenceNumber sn
= new SequenceNumber();
③ 3
个线程共享sn
,各自产生序列号
TestClient t1 = new TestClient(sn
);
TestClient t2 = new TestClient(sn
);
TestClient t3 = new TestClient(sn
);
t1.start();
t2.start();
t3.start();
}
private static class TestClient extends Thread
{
private SequenceNumber sn;
public TestClient(SequenceNumber sn) {
this.sn = sn;
}
public void run()
{
for (int i = 0; i < 3; i++) {④
每个线程打出3
个序列值
System.out.println("thread["+Thread.currentThread().getName()+
"] sn["+sn.getNextNum()
+"]");
}
}
}
}
通常我们通过匿名内部类的方式定义ThreadLocal
的子类,提供初始的变量值,如例子中①
处所示。TestClient
线程产生一组序列号, 在③
处,我们生成3
个TestClient
,它们共享同一个SequenceNumber
实例。运行以上代码,在控制台上输出以下的结果:
thread[Thread-2] sn[1]
thread[Thread-0] sn[1]
thread[Thread-1] sn[1]
thread[Thread-2] sn[2]
thread[Thread-0] sn[2]
thread[Thread-1] sn[2]
thread[Thread-2] sn[3]
thread[Thread-0] sn[3]
thread[Thread-1] sn[3]
考察输出的结果信息,我们发现每个线程所产生的序号虽然都共享同一个SequenceNumber
实例,但它们并没有发生相互干扰的情况,而是各自产生独立的序列号,这是因为我们通过ThreadLocal
为每一个线程提供了单独的副本。
Thread
同步机制的比较
ThreadLocal
和线程同步机制相比有什么优势呢?ThreadLocal
和线程同步机制都是为了解决多线程中相同变量的访问冲突问题。
在同步机制中,通过对象的锁机制保证同一时间只有一个线程访问变量。这时该变量是多个线程共享的,使用同步机制要求程序慎密地分析什么时候对变量进行读写,什么时候需要锁定某个对象,什么时候释放对象锁等繁杂的问题,程序设计和编写难度相对较大。
而ThreadLocal
则从另一个角度来解决多线程的并发访问。ThreadLocal
会为每一个线程提供一个独立的变量副本,从而隔离了多个线 程对数据的访问冲突。因为每一个线程都拥有自己的变量副本,从而也就没有必要对该变量进行同步了。ThreadLocal
提供了线程安全的共享对象,在编 写多线程代码时,可以把不安全的变量封装进ThreadLocal
。
由于ThreadLocal
中可以持有任何类型的对象,低版本JDK
所提供的get()
返回的是Object
对象,需要强制类型转换。但JDK 5.0
通过泛型很好的解决了这个问题,在一定程度地简化ThreadLocal
的使用,代码清单 9 2
就使用了JDK 5.0
新的ThreadLocal<T>
版本。
概括起来说,对于多线程资源共享的问题,同步机制采用了“
以时间换空间”
的方式,而ThreadLocal
采用了“
以空间换时间”
的方式。前者仅提供一份变量,让不同的线程排队访问,而后者为每一个线程都提供了一份变量,因此可以同时访问而互不影响。
Spring
使用ThreadLocal
解决线程安全问题
我们知道在一般情况下,只有无状态的Bean
才可以在多线程环境下共享,在Spring
中,绝大部分Bean
都可以声明为singleton
作用 域。就是因为Spring
对一些Bean
(如RequestContextHolder
、 TransactionSynchronizationManager
、LocaleContextHolder
等)中非线程安全状态采用 ThreadLocal
进行处理,让它们也成为线程安全的状态,因为有状态的Bean
就可以在多线程中共享了。
一般的Web
应用划分为展现层、服务层和持久层三个层次,在不同的层中编写对应的逻辑,下层通过接口向上层开放功能调用。在一般情况下,从接收请求到返回响应所经过的所有程序调用都同属于一个线程,如图所示:
图
1
同一线程贯通三层
这样你就可以根据需要,将一些非线程安全的变量以ThreadLocal
存放,在同一次请求响应的调用线程中,所有关联的对象引用到的都是同一个变量。
下面的实例能够体现Spring
对有状态Bean
的改造思路:
代码清单3 TopicDao
:非线程安全
public class TopicDao {
private Connection conn;①
一个非线程安全的变量
public void addTopic(){
Statement stat = conn.createStatement();②
引用非线程安全变量
…
}
}
由于①
处的conn
是成员变量,因为addTopic()
方法是非线程安全的,必须在使用时创建一个新TopicDao
实例(非singleton
)。下面使用ThreadLocal
对conn
这个非线程安全的“
状态”
进行改造:
代码清单4 TopicDao
:线程安全
import java.sql.Connection;
import java.sql.Statement;
public class TopicDao {
①
使用ThreadLocal
保存Connection
变量
private static ThreadLocal<Connection>
connThreadLocal = new
ThreadLocal<Connection>();
public static Connection getConnection(){
②
如果connThreadLocal
没有本线程对应的Connection
创建一个新的Connection
,
并将其保存到线程本地变量中。
if (connThreadLocal.get() == null) {
Connection conn = ConnectionManager.getConnection();
connThreadLocal.set(conn);
return conn;
}else{
return connThreadLocal.get();③
直接返回线程本地变量
}
}
public void addTopic() {
④
从ThreadLocal
中获取线程对应的Connection
Statement stat = getConnection().createStatement();
}
}
不同的线程在使用TopicDao
时,先判断connThreadLocal.get()
是否是null
,如果是null
,则说明当前线程还没有对 应的Connection
对象,这时创建一个Connection
对象并添加到本地线程变量中;如果不为null
,则说明当前的线程已经拥有了 Connection
对象,直接使用就可以了。这样,就保证了不同的线程使用线程相关的Connection
,而不会使用其它线程的 Connection
。因此,这个TopicDao
就可以做到singleton
共享了。
当然,这个例子本身很粗糙,将Connection
的ThreadLocal
直接放在DAO
只能做到本DAO
的多个方法共享Connection
时
不发生线程安全问题,但无法和其它DAO
共用同一个Connection
,要做到同一事务多DAO
共享同一Connection
,必须在一个共同的外部类 使用ThreadLocal
保存Connection
。
小结
ThreadLocal
是解决线程安全问题一个很好的思路,它通过为每个线程提供一个独立的变量副本解决了变量并发访问的冲突问题。在很多情况
下,ThreadLocal
比直接使用synchronized
同步机制解决线程安全问题更简单,更方便,且结果程序拥有更高的并发性。
- 大小: 16.9 KB
分享到:
相关推荐
关于线程变量ThreadLocal的介绍以及说明. 关于线程变量ThreadLocal的介绍以及说明. 关于线程变量ThreadLocal的介绍以及说明. 关于线程变量ThreadLocal的介绍以及说明. 关于线程变量ThreadLocal的介绍以及说明. ...
主要介绍ThreadLocal的原理,实例分析以及注意事项
主要介绍了彻底理解Java 中的ThreadLocal的相关资料,需要的朋友可以参考下
主要介绍了Android 中 ThreadLocal使用示例的相关资料,这里提供示例代码帮助大家学习理解这部分内容,需要的朋友可以参考下
主要介绍了Java ThreadLocal用法,结合实例形式详细分析了ThreadLocal线程局部变量相关原理、定义与使用方法,需要的朋友可以参考下
主要介绍了Android 中ThreadLocal的深入理解的相关资料,希望通过本文能帮助到大家,让大家理解应用ThreadLocal,需要的朋友可以参考下
ThreadLocal翻译成中文比较准确的叫法应该是:线程局部变量。使用这个工具类可以很简洁地编写出优美的多线程程序。接下来通过本文给大家介绍Java中的ThreadLocal,需要的朋友可以参考下
主要介绍了java 中ThreadLocal 的正确用法的相关资料,需要的朋友可以参考下
主要介绍了Java 并发编程之ThreadLocal详解及实例的相关资料,需要的朋友可以参考下
主要介绍了从面试中的问题分析ThreadLocal,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,下面我们来一起学习一下吧
主要介绍了ThreadLocal使用案例分析,需要的朋友可以参考下
主要介绍了Java ThreadLocal类应用,结合具体案例形式分析了java ThreadLocal类的功能、原理、用法及相关操作注意事项,需要的朋友可以参考下
主要为大家详细介绍了java ThreadLocal的使用案例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
主要介绍了关于ThreadLocal对request和response的用法说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
目录SimpleDateFormat诡异bug复现SimpleDateFormat诡异bug字符串日期转Date日期(parse)Date...介绍ThreadLocal使用demoThreadLocal源码探索ThreadLocal注意事项使用ThreadLocal解决SimpleDateFormat线程安全问题总结...
主要介绍了ThreadLocal原理及内存泄漏原因,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
主要介绍了深入理解ThreadLocal工作原理及使用示例,涉及ThreadLocal<T> 简介和使用示例及ThreadLocal的原理等相关内容,具有一定参考价值,需要的朋友可以了解下。
主要介绍了java 中ThreadLocal实例分析的相关资料,需要的朋友可以参考下
主要介绍了Java ThreadLocal的设计理念与作用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
目录一、背景介绍二、TestNG多线程详解2.1 TestNG多线程实现2.2 TestNG多线程效果演示三、ThreadLocal3.1 ThreadLocal概念3.2 具体实现 一、背景介绍 在使用Selenium+TestNG做WebUI自动化过程中,为了能够加快Web...