使用ThreadLocal对象有两个主要的功能:
- 实现对同一个实例化对象不同线程属性值得分离;
- 同一个会话(线程)中对象值的传递。
以下两个类用来解释ThreadLocal对象的两个功能,MyThreadLocal:单例类,具有一个ThreadLocal对象的属性;TestRunnable:实现线程接口Runnable,在run方法中,获取MyThreadLocal对象的单例->获取使用ThreadLocal对象存储的值->设置ThreadLocal对象的新值。
为了能够直观的看出ThreadLocal对象值的改变,使用两个线程,并且设置ThreadLocal对象的值后暂停线程1s。
以下是具体代码:
MyThreadLocal类:
public class MyThreadLocal { private ThreadLocal<Integer> flag = new ThreadLocal<Integer>() ; private static MyThreadLocal local = new MyThreadLocal() ; private MyThreadLocal(){ } /** * 通过单例获取一个实例化的对象 * @return */ public static MyThreadLocal getInstance(){ return local ; } public void setFlagValue(int num){ flag.remove() ; flag.set(num) ; } public int getFlagValue(){ if(flag.get()==null){ return 0 ; } return flag.get() ; } public ThreadLocal<Integer> getFlag() { return flag; } public void setFlag(ThreadLocal<Integer> flag) { this.flag = flag; } }
TestRunnable类:
public class TestRunnable implements Runnable { private String name ; public TestRunnable(String name){ this.name = name ; } public String getName() { return name; } public void setName(String name) { this.name = name; } /** * 在设置值之前首先输出通过ThreadLocal对象保存的值,设置新值后再经过sleep, * 使其他的线程能够操作同一个对象的属性。 */ @Override public void run() { MyThreadLocal myThreadLocal = MyThreadLocal.getInstance() ; for (int i = 0; i < 10; i++) { System.out.println(name+" 前-> "+myThreadLocal.getFlagValue()) ; if(name.indexOf("1")>-1){ myThreadLocal.setFlagValue(i) ; } try { if(name.indexOf("1")>-1&&i%2==0){ Thread.currentThread().sleep(1000) ; } if(name.indexOf("2")>-1&&i%2==1){ Thread.currentThread().sleep(1000) ; } } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(name+" 后-> "+myThreadLocal.getFlagValue()) ; } } public static void main(String[] args) { Thread thread1 = new Thread(new TestRunnable("thread1")) ; Thread thread2 = new Thread(new TestRunnable("thread2")) ; thread1.start(); thread2.start(); } }
运行一次的结果如下:
thread1 前-> 0
thread2 前-> 0
thread2 后-> 0
thread2 前-> 0
thread2 后-> 0
thread1 后-> 0
thread2 前-> 0
thread1 前-> 0
thread2 后-> 0
thread1 后-> 1
thread2 前-> 0
thread1 前-> 1
thread2 后-> 0
thread1 后-> 2
thread1 前-> 2
thread2 前-> 0
thread1 后-> 3
thread2 后-> 0
thread1 前-> 3
thread2 前-> 0
thread2 后-> 0
thread2 前-> 0
thread2 后-> 0
thread2 前-> 0
thread1 后-> 4
thread1 前-> 4
thread1 后-> 5
thread1 前-> 5
thread1 后-> 6
thread1 前-> 6
thread2 后-> 0
thread2 前-> 0
thread2 后-> 0
thread2 前-> 0
thread1 后-> 7
thread1 前-> 7
thread2 后-> 0
thread1 后-> 8
thread1 前-> 8
thread1 后-> 9
由结果中可以看出:通过单例类设置和获取的值是某个线程自己的值,不同的线程之间没有共享ThreadLocal对象的值。
另参考了http://www.iteye.com/topic/103804如下内容:
首先,ThreadLocal 不是用来解决共享对象的多线程访问问题的,一般情况下,通过ThreadLocal.set() 到线程中的对象是该线程自己使用的对象,其他线程是不需要访问的,也访问不到的。各个线程中访问的是不同的对象。
另外,说ThreadLocal使得各线程能够保持各自独立的一个对象,并不是通过ThreadLocal.set()来实现的,而是通过每个线程中的new 对象 的操作来创建的对象,每个线程创建一个,不是什么对象的拷贝或副本。通 过ThreadLocal.set()将这个新创建的对象的引用保存到各线程的自己的一个map中,每个线程都有这样一个map,执行 ThreadLocal.get()时,各线程从自己的map中取出放进去的对象,因此取出来的是各自自己线程中的对象,ThreadLocal实例是作 为map的key来使用的。
如果ThreadLocal.set()进去的东西本来就是多个线程共享的同一个对象,那么多个线程的ThreadLocal.get()取得的还是这个共享对象本身,还是有并发访问问题。
下面来看一个hibernate中典型的ThreadLocal的应用:
private static final ThreadLocal threadSession = new ThreadLocal(); public static Session getSession() throws InfrastructureException { Session s = (Session) threadSession.get(); try { if (s == null) { s = getSessionFactory().openSession(); threadSession.set(s); } } catch (HibernateException ex) { throw new InfrastructureException(ex); } return s; }
可以看到在getSession()方法中,首先判断当前线程中有没有放进去session,如果还没有,那么通过 sessionFactory().openSession()来创建一个session,再将session set到线程中,实际是放到当前线程的ThreadLocalMap这个map中,这时,对于这个session的唯一引用就是当前线程中的那个 ThreadLocalMap(下面会讲到),而threadSession作为这个值的key,要取得这个session可以通过 threadSession.get()来得到,里面执行的操作实际是先取得当前线程中的ThreadLocalMap,然后将 threadSession作为key将对应的值取出。这个session相当于线程的私有变量,而不是public的。
显然,其他线程中是取不到这个session的,他们也只能取到自己的ThreadLocalMap中的东西。要是session是多个线程共享使用的,那还不乱套了。
试想如果不用ThreadLocal怎么来实现呢?可能就要在action中创建session,然后把session一个个传到service 和dao中,这可够麻烦的。或者可以自己定义一个静态的map,将当前thread作为key,创建的session作为值,put到map中,应该也 行,这也是一般人的想法,但事实上,ThreadLocal的实现刚好相反,它是在每个线程中有一个map,而将ThreadLocal实例作为key, 这样每个map中的项数很少,而且当线程销毁时相应的东西也一起销毁了,不知道除了这些还有什么其他的好处。 以ThreadLocal实例作为key,要保持的对象作为值,设置到当前线程的ThreadLocalMap 中。
总之,ThreadLocal不是用来解决对象共享访问问题的,而主要是提供了保持对象的方法和避免参数传递的方便的对象访问方式。归纳了两点:
1。每个线程中都有一个自己的ThreadLocalMap类对象,可以将线程自己的对象保持到其中,各管各的,线程可以正确的访问到自己的对象。
2。将一个共用的ThreadLocal静态实例作为key,将不同对象的引用保存到不同线程的ThreadLocalMap中,然后在线程执行 的各处通过这个静态ThreadLocal实例的get()方法取得自己线程保存的那个对象,避免了将这个对象作为参数传递的麻烦。
当然如果要把本来线程共享的对象通过ThreadLocal.set()放到线程中也可以,可以实现避免参数传递的访问方式,但是要注意 get()到的是那同一个共享对象,并发访问问题要靠其他手段来解决。但一般来说线程共享的对象通过设置为某类的静态变量就可以实现方便的访问了,似乎没必要放到线程中。
ThreadLocal的应用场合,我觉得最适合的是按线程多实例(每个线程对应一个实例)的对象的访问,并且这个对象很多地方都要用到。
相关推荐
**ThreadLocal的使用方法:** 1. **创建ThreadLocal实例:** 首先,你需要创建一个ThreadLocal类型的实例,这将作为你在每个线程中存储值的容器。 ```java ThreadLocal<String> threadLocal = new ThreadLocal();...
由于ThreadLocal变量是存储在线程的ThreadLocalMap中,如果线程长时间运行并且不清理ThreadLocal,当ThreadLocal对象被垃圾收集时,其在ThreadLocalMap中的引用将变为"幽灵引用"(弱引用),导致内存泄漏。...
在Java多线程编程中,`ThreadLocal`是一个非常重要的工具类,它提供了一种在每个线程内部存储线程私有实例的方法。通常情况下,当多个线程共享某个变量时,可能会引发线程安全问题。而`ThreadLocal`则提供了另一种...
这意味着即使两个不同的线程分别在同一个ThreadLocal对象上设置不同的值,它们也无法访问到对方在该ThreadLocal对象上设置的值。 为了简化类型转换,我们可以为ThreadLocal指定泛型类型。例如,如果我们只想存储...
Hibernate框架在处理数据库操作时,使用ThreadLocal来管理Session,确保每个线程都有自己独立的Session对象。这样,即使多个线程并发执行,每个线程也不会干扰其他线程的Session,避免了数据错乱的问题。例如,...
ThreadLocal的使用方法是创建一个ThreadLocal实例,然后通过其set()方法设置线程局部变量,get()方法获取当前线程的该变量值。需要注意的是,ThreadLocal不是线程安全的,它只是保证了线程内部的隔离性,但不负责...
2. **请求上下文**:在Web应用中,可以使用`ThreadLocal`保存HttpServletRequest或Session对象,方便在同一个请求处理链路中访问。 3. **线程状态**:记录线程执行过程中的状态,例如事务管理、计数器等。 然而,...
接着,使用ThreadLocal对象作为key,传入的value作为value,将这对键值对存入ThreadLocalMap中(map.set(this, value);)。 当其他线程访问同一个ThreadLocal实例时,它们会拥有自己的ThreadLocalMap,所以不会影响...
这个副本存储在线程的ThreadLocalMap中,这个Map是由Thread类维护的,键是ThreadLocal对象,值是线程局部变量的副本。 使用ThreadLocal的步骤如下: 1. 创建ThreadLocal实例。 2. 使用`set`方法设置线程局部变量的...
下面将对 ThreadLocal 的原理、实现机制、使用方法等进行深入分析。 什么是 ThreadLocal? ThreadLocal 翻译过来就是本地线程,但是直接这么翻译很难理解 ThreadLocal 的作用。如果换一种说法,可以称为线程本地...
当线程存活时间过长,而ThreadLocal对象已经不再使用,如果没有及时调用`remove()`,则其在ThreadLocalMap中的引用无法被垃圾回收,进而可能导致内存泄漏。因此,对于长时间运行的服务,应特别注意ThreadLocal的生命...
早在 JDK 1.2 的时代,java.lang.ThreadLocal 就诞生了,它是为了解决多线程并发问题而设计的,只不过设计得有些难用,所以至今没有得到广泛使用。其实它还是挺有用的,不相信的话,我们一起来看看这个例子吧。 一个...
这意味着ThreadLocal对象是弱引用的,如果ThreadLocal变量没有其他强引用指向它,那么在垃圾收集期间,即使ThreadLocalMap中仍然有引用,该ThreadLocal对象也会被回收。Entry中的value字段存储了与ThreadLocal关联的...
ThreadLocal是Java中一...然而,需要注意的是,如果不再使用ThreadLocal变量,应适当调用`remove()`方法,以防止内存泄漏,因为线程结束后,ThreadLocal的引用可能不会立即消失,导致内存中的变量副本无法被垃圾收集。
Java并发编程中,ThreadLocal是一个非常重要的类,它是解决线程安全问题的一个工具。...ThreadLocal是一个功能强大但又需要仔细使用的工具,掌握其原理和正确使用方法对于编写高性能和高可靠性的Java应用程序至关重要。
如果在一个类的两个方法都使用`synchronized`,并且调用这些方法的线程持有相同的对象引用,那么这两个方法不能同时执行。 2. **ThreadLocal** ThreadLocal是一种线程局部变量,它为每个线程都创建了一个独立的...
ThreadLocal的使用方法通常是创建一个ThreadLocal实例,然后在需要的地方通过它的`set()`方法设置线程局部变量的值,通过`get()`方法获取该变量的副本。在生命周期结束后,通常需要调用`remove()`方法清除线程的副本...
- **弱引用**:ThreadLocalMap的键使用的是弱引用,当ThreadLocal变量不再被引用时,垃圾收集器可以回收ThreadLocal对象,但其在ThreadLocalMap中的引用不会立即被移除,以防止导致内存泄漏。 - **线程生命周期**:...