`
zhyxfancy
  • 浏览: 52044 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

ThreadLocal 学习

阅读更多
关于ThreadLocal的用法在进行了诸多的研究分析之后,终于有所斩获,掌握了其本质的若干用法,先总结如下:

在对运行同一段代码的多线程程序中,用以实现同一个线程之内同一个变量的同步,避免其多个线程之间彼此的干扰。效率相对于synchronized而言,有相当的提升。
本质上讲,其就是实现单一线程的私有变量
关于其实现,摘引klyuan(javaeye)上的例子,如下,很好的说明了ThreadLocal的实现方式,当然你也可以参考JDK源代码中的实现。本质上,其就是一个同步的HashMap而已,可以为线程,value为所定义的局部变量。

public class ThreadLocal 

 private Map values = Collections.synchronizedMap(new HashMap()); 
 public Object get() 
 { 
  Thread curThread = Thread.currentThread();  
  Object o = values.get(curThread);  
  if (o == null && !values.containsKey(curThread)) 
  { 
   o = initialValue(); 
   values.put(curThread, o);  
  } 
  return o;  
 } 
 
 public void set(Object newValue) 
 { 
  values.put(Thread.currentThread(), newValue); 
 } 
 
 public Object initialValue() 
 { 
  return null;  
 } 

应用分析,一个同事将一段代码给我,说其中关于ThreadLocal的set(null)这行代码,如果注释掉,会出现什么情况,详细代码如下:

我们可以看到有2行关于local和local_tr的set(null)的方法如果被注释掉的情况,我们可以推测会出现什么样的情况。

/**
  *  Import Package missing for space
  **/
public class HibernateUtil {
         private static SessionFactory sf;
         private static ThreadLocal local = new ThreadLocal();
         private static ThreadLocal local_tr = new ThreadLocal();
      
       static{
             Configuration conf = new Configuration();
             conf = conf.configure("/hibernate.cfg.xml");
             sf = conf.buildSessionFactory();
        }

       public static Session getSession(){
             System.out.println("threadLocalid=="+local);
             Session session = null;
        if(local.get()==null){
             session = sf.openSession();
             System.out.println("---------session is created------------");
              local.set(session);
         }else{
            session = (Session)local.get();
       }
          return session;
      }
     
     public static void begin(){
           Session session = (Session)local.get();
          Transaction ts = null;
          if(local_tr.get()==null){
                 ts = session.beginTransaction();
                 local_tr.set(ts);
          }
        }

        public static void commit(){
            Transaction tr = (Transaction)local_tr.get();
            tr.commit();
            local_tr.set(null); // this line is removed,then what will happen?
          System.out.println("---------transaction is commited------------");
        }

       public static void rollback(){
                Transaction tr = (Transaction)local_tr.get();
                 tr.rollback();
                System.out.println("---------transaction is rollbacked------------");
         }

        public static void close(){
                Session session = (Session)local.get();
                if(session!=null){
                        session.close();
                        System.out.println("---------session is destroyed------------");
         }
             local.set(null);  // this line is removed, then what will happen?
      }
}



如果基于local(Session的局部变量)不进行set(null)的话,会出现所有的thread都基于获取同一个session的情况。

如果local_tr(Transaction的局部变量)不进行set(null)的话,对于基于同一个session的线程而言,其只可以进行一次的事务提交,后续都将无法进行,因为每次进行之前会进行局部变量的有效判断。

总结: ThreadLocal是一个有效的多线程局部变量工具,当然也是一把双刃剑,用好的话可以制敌,反之则可能祸己。要用好ThreadLocal,我们只需记住其本质:线程局部变量;其实现的本质是一个基于同步的HashMap就可以了。每一个线程都是用于该变量的一个独立副本。
分享到:
评论
2 楼 837062099 2012-03-10  
引用
 private Map values = Collections.synchronizedMap(new HashMap()); 

老版本的jdk,jdk1.6里是基于ThreadLocalMap实现的,效率更快。
1 楼 greens.leaf 2009-01-06  
Webwork/Xwork 中 ServletActionContext.getRequest() 就是基于 ThreadLocal实现的,可以参考源码。

相关推荐

    ThreadLocal详解.md

    学习ThreadLocal,了解其中的原理,以及学习其中的优点!避免坑点!!

    设计模式及ThreadLocal资料

    设计模式及ThreadLocal详细讲解资料,想要学习java或者提升自己技术的同学可以下载观看

    Quartz-ThreadLocal.rar

    本地学习练习demo的eclipse工作空间:主要包括多线程的相关demo以及quartz调度的简单实现和其他java基础的demo练习

    java核心知识点学习----多线程间的数据共享和对象独立,ThreadLocal详解.pdf

    java核心知识点学习----多线程间的数据共享和对象独立,ThreadLocal详解.pdf

    Android 中 ThreadLocal使用示例

    主要介绍了Android 中 ThreadLocal使用示例的相关资料,这里提供示例代码帮助大家学习理解这部分内容,需要的朋友可以参考下

    ThreadLocal,你真的了解吗?

    在学习ThreadLocal之前,先了解一下java中的四种引用,大厂面试的时候,面试官一般都是先问四种引用,然后过渡到ThreadLocal。 2.Java中的引用类型 从Java SE2开始,就提供了四种类型的引用:强引用、软引用、弱引用...

    从面试中的问题分析ThreadLocal

    主要介绍了从面试中的问题分析ThreadLocal,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,下面我们来一起学习一下吧

    Java ThreadLocal的设计理念与作用

    主要介绍了Java ThreadLocal的设计理念与作用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

    ThreadLocal原理及内存泄漏原因

    主要介绍了ThreadLocal原理及内存泄漏原因,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

    Java并发编程学习之ThreadLocal源码详析

    主要给大家介绍了关于Java并发编程学习之源码分析ThreadLocal的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

    有关ThreadLocal的面试题你真的懂了吗

    主要介绍了面试题ThreadLocal,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

    Java单线程ThreadLocal串值问题解决方案

    主要介绍了Java单线程ThreadLocal串值问题解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

    京东一面:说出ThreadLocal的使用场景及使用方式.zip

    计算机技术、IT咨询、人工智能AI理论介绍,学习参考资料 计算机技术、IT咨询、人工智能AI理论介绍,学习参考资料 计算机技术、IT咨询、人工智能AI理论介绍,学习参考资料 计算机技术、IT咨询、人工智能AI理论介绍,...

    深入浅出的学习Java ThreadLocal

    本文会基于实际场景介绍ThreadLocal如何使用以及内部实现机制。具有很好的参考价值,下面跟着小编一起来看下吧

    深入学习java ThreadLocal的源码知识

    ThreadLocal是一个本地线程副本变量工具类。主要用于将私有线程和该线程存放的副本对象做一个映射,各个线程之间的变量互不干扰,特别适用于各个线程依赖不通的变量值完成操作的场景。下面我们来详细了解一下它吧

    java学习整理文档.docx

    java 学习整理文档 Spring框架并没有对单例bean进行任何多线程的封装处理。关于单例bean的线程安全和并发问题需要开发者自行去搞定 最浅显的解决办法就是将多态bean的作用域由“singleton”变更为“prototype”或者...

    18 线程作用域内共享变量—深入解析ThreadLocal.pdf

    Java并发编程学习宝典(漫画版),Java并发编程学习宝典(漫画版)Java并发编程学习宝典(漫画版)Java并发编程学习宝典(漫画版)Java并发编程学习宝典(漫画版)Java并发编程学习宝典(漫画版)Java并发编程学习...

    深入理解 Java 之 ThreadLocal 工作原理1

    声明仅作学习。如有不适,请告知。清晰的看到一个线程Thread中存在一个ThreadLocalMap,ThreadLocalMap中的key对应ThreadLo

    53.线程间的通信-join方法-ThreadLocal类.mp4

    在学习Java过程中,自己收集了很多的Java的学习资料,分享给大家,有需要的欢迎下载,希望对大家有用,一起学习,一起进步。

Global site tag (gtag.js) - Google Analytics