ThreadLocal是什么-java ThreadLocal学习笔记

2019-03-12 23:36|来源: 网络

ThreadLocal是什么

ThreadLocal是什么呢?其实ThreadLocal并非是一个线程的本地实现版本,它并不是一个Thread,而是thread local variable(线程局部变量)。也许把它命名为ThreadLocalVar更加合适。线程局部变量(ThreadLocal)其实的功用非常简单,就是为每一个使用该变量的线程都提供一个变量值的副本,是每一个线程都可以独立地改变自己的副本,而不会和其它线程的副本冲突。从线程的角度看,就好像每一个线程都完全拥有该变量。线程局部变量并不是Java的新发明,在其它的一些语言编译器实现(如IBM XL FORTRAN)中,它在语言的层次提供了直接的支持。因为Java中没有提供在语言层次的直接支持,而是提供了一个ThreadLocal的类来提供支持,所以,在Java中编写线程局部变量的代码相对比较笨拙,这也许是线程局部变量没有在Java中得到很好的普及的一个原因吧。

ThreadLocal的设计

首先看看ThreadLocal的接口:
Object get() ;                         // 返回当前线程的线程局部变量副本

protected Object initialValue();       // 返回该线程局部变量的当前线程的初始值
void set(Object value);                // 设置当前线程的线程局部变量副本的值

ThreadLocal有3个方法,其中值得注意的是initialValue(),该方法是一个protected的方法,显然是为了子类重写而特意实现的。该方法返回当前线程在该线程局部变量的初始值,这个方法是一个延迟调用方法,在一个线程第1次调用get()或者set(Object)时才执行,并且仅执行1次。ThreadLocal中的确实实现直接返回一个null:

protected Object initialValue() { return null; }

ThreadLocal是如何做到为每一个线程维护变量的副本的呢?其实实现的思路很简单,在ThreadLocal类中有一个Map,用于存储每一个线程的变量的副本。比如下面的示例实现:
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;
   }
  }

  当然,这并不是一个工业强度的实现,但JDK中的ThreadLocal的实现总体思路也类似于此。


转自网络


相关问答

更多
  • SequenceNumber package com.baobaotao.basic; public class SequenceNumber { ①通过匿名内部类覆盖ThreadLocal的initialValue()方法,指定初始值 private static ThreadLocal seqNum = new ThreadLocal (){ public Integer initialValue(){ return 0; } }; ②获取下一个序列值 public int getNextNum(){ ...
  • 从如下8点来讲解一下: 1.ThreadLocal用来解决多线程程序的并发问题 2.ThreadLocal并不是一个Thread,而是Thread的局部变量,当使用ThreadLocal维护变量时,ThreadLocal为每个使用该变量的线程提供独立的变量副本,所以每个线程都 可以独立地改变自己的副本,而不会影响其它线程所对应的副本. 3.从线程的角度看,目标变量就象是线程的本地变量,这也是类名中“Local”所要表达的意思。 4.线程局部变量并不是Java的新发明,Java没有提供在语言级支持(语法上), ...
  • 这是安全的,因为getMap返回给定(即当前)线程的映射。 没有其他线程会与此混淆。 所以它实际上getMap的实现,以确保任何线程都可以 - 并且据我所知,它只是委托给Thread对象中的一个字段。 我不清楚getMap是否传递了除当前线程之外的任何线程 - 如果是,那可能是getMap棘手的 - 但我怀疑它们都是经过仔细编写以确保不是问题:) It's safe because getMap returns the map for the given (i.e. current) thread. No ...
  • 这里的所有答案都是正确的,但有点令人失望,因为它们ThreadLocal了解ThreadLocal的实现方式。 我只是看看ThreadLocal的源代码,并对它的实现感到ThreadLocal 。 天真的实现 如果我要求您实现一个给定javadoc中描述的API的ThreadLocal类,那么你会怎么做? 最初的实现可能是使用Thread.currentThread()作为其键的ConcurrentHashMap 。 这将工作得很好,但确实有一些缺点。 线程争用 - Concurr ...
  • 因为如果它是一个实例级别的字段,那么它实际上是“Per Thread - Per Instance”,而不仅仅是一个保证的“Per Thread”。 这通常不是您要查找的语义。 通常它持有类似于用户对话,Web请求等范围的对象的东西。您不希望它们也被分配到类的实例。 一个Web请求=>一个持久性会话。 每个对象没有一个Web请求=>一个持久性会话。 Because if it were an instance level field, then it would actually be "Per Threa ...
  • 线程本地缓存副本很好,因为它更简单,更有效。 一个缓存副本,你不确定它是本地线程可能是一个问题。 单身定义意味着只能有一个。 我不会把你作为一个单身人物所拥有的,你每个线程都有一个。 我会在它的构造函数中初始化()你的线程局部对象。 BTW你的ThreadLocal应该是private static final我怀疑。 A thread local cached copy is fine as it is simpler and more efficient. A cached copy where you ...
  • 通过使用ThreadLocal,可以创建与线程一样多的变量,而无需进一步检查。 但请记住,存储本身并不能保证线程安全。 您必须确保存储在本地存储中的每个对象仅在该线程中使用! 如果您手动克隆对象,则每次使用对象时都必须克隆对象,或者检查我们是哪个线程,然后克隆。 此外 - 克隆操作是否线程安全? 如果两个不同的线程试图克隆一个对象会发生什么? 我其实不知道,但我认为这不是一个好的做法。 By using ThreadLocal you create as many variables as there ar ...
  • ThreadLocal的目的是这些字段不必是原子的 - 不需要将它们的值与集中式内存同步。 它们是线程本地的,只存在于线程本地存储器中。 为什么我会想要使用ThreadLocal而不是AtomicLong或AtomicInteger? ThreadLocal对于存储某个东西的每个线程副本非常有用。 例如SimpleDateFormat不幸是不可重入的。 private final ThreadLocal threadLocal = new ThreadLocal
  • 您的代码的问题是您将保持文件指针打开。 您可以将对象存储在ThreadLocal中,但不存储诸如文件句柄,数据库连接或其他需要关闭的资源等资源。 ThreadLocal将超出范围,因此一旦Thread完成就会被垃圾收集。 在您的情况下,我将存储您从ThreadLocal中的InputStream获取的内容,但不存储流本身。 如果您想确保某些内容始终可用,请使用Singleton模式。 请注意Java EE环境中模式的局限性。 The problem with your code is that you wi ...
  • ThreadLocal local = new ThreadLocal(); local.set("String"); System.out.println(local.get()); 您需要在ThreadLocalVariable中设置一些内容然后检索它。 最初,ThreadLocal为空。 ThreadLocal local = new ThreadLocal(); local.set("String"); System.out.prin ...