ThreadLocal线程局部变量产生,应用及原理


new一个map,线程号做key,共享变量做value也是可以的

private Map<Thread,long> map =new hashmap<>();
map.put(Thead.currentThread,1)
map.get(Thead.currentThread)

ThreadLocal

ThreadLocal为解决多线程程序的并发问题提供了一种新的思路。使用这个工具类可以很简洁地编写出优美的多线程程序,ThreadLocal并不是一个Thread,而是Thread的局部变量。(它可以为每一个线程维护变量的副本-------[实现原理:在ThreadLocal类中定义了一个ThreadLocalMap,每一个Thread中都有一个该类型的变量——threadLocals——用于存储每一个线程的变量副本,Map中元素的键为线程对象,而值对应线程的变量副本。
在这里插入图片描述
类似单例类TransactionSynchronizationManager,

RequestContextHolder中就是通过ThreadLocal保存各自线程变量的副本,这样就不需要重新创建类

Thread curThread = Thread.currentThread();
private Map values = Collections.synchronizedMap(new HashMap());
values.put(curThread, (Object)values.get(curThread));

/**
*说白了,就是将当前线程运行产生的数据(变量)依次放入ThreadLocal的Map里面,
需要的时候,传入key值,从threadlocal的map里取出即可。在多线程环境下,
由于采用的是局部变量的方式存取值,不加锁。因此高效且安全。因此应用非常广泛。比如
*/

//########################hibernate的ssion操作
1. public static final ThreadLocal session = new ThreadLocal();
2. public static Session currentSession() {
3. Session s = (Session)session.get();
4. //open a new session,if this session has none
5. if(s == null){
6. s = sessionFactory.openSession();
7. session.set(s);
8. }
return s;
9. }
我们逐行分析
1。 初始化一个ThreadLocal对象,ThreadLocal有三个成员方法 get()set()initialvalue()。
如果不初始化initialvalue,则initialvalue返回null。
3。session的get根据当前线程返回其对应的线程内部变量,也就是我们需要的net.sf.hibernate.Session(相当于对应每个数据库连接).多线程情况下共享数据库链接是不安全的。ThreadLocal保证了每个线程都有自己的s(数据库连接)。
5。如果是该线程初次访问,自然,s(数据库连接)会是null,接着创建一个Session,具体就是行66。创建一个数据库连接实例 s
7。保存该数据库连接s到ThreadLocal中。
8。如果当前线程已经访问过数据库了,则从session中get()就可以获取该线程上次获取过的连接实例。


//####################我想给jdbc的线程初始一个特殊值(数据源like c3p0,druid..)
private static class JDBCContextThreadLocal extends ThreadLocal {
public javax.sql.DataSource ds;
public JDBCContextThreadLocal(javax.sql.DataSource ds)
{
this.ds=ds;
}
protected synchronized Object initialValue() {
return new JDBCContext(ds);
}}

])由于Java中编写线程局部变量的代码相对来说要笨拙一些,因此造成线程局部变量没有在Java开发者中得到很好的普及。----ThreadLocal的出现是为了解决这个问题的在jdk1.5以后它是一个泛型类 class ThreadLocal
它主要有四个方法:
1.public void remove()
将当前线程局部变量的值删除,目的是为了减少内存的占用.当线程结束后,对应该线程的局部变量将自动被垃圾回收,所以显式调用该方法(不是必须的操作)可以加快内存回收的速度。
2.protected Object initialValue()
返回该线程局部变量的初始值,该方法是一个protected的方法,显然是为了让子类覆盖而设计的。这个方法是一个延迟调用方法,在线程第1次调用get()或set(Object)时才执行,并且仅执行1次。ThreadLocal中的缺省实现直接返回一个null。

总结

ThreadLocal是解决线程安全问题一个很好的思路,它通过为每个线程提供一个独立的变量副本解决了变量并发访问的冲突问题。在很多情况下,ThreadLocal比直接使用synchronized同步机制解决线程安全问题更简单,更方便,且结果程序拥有更高的并发性。

转载自:

百度百科:https://baike.baidu.com/item/ThreadLocal/4915311?fr=aladdin
为什么面试经常拿ThreadLocal考验求职者了

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值