返回顶部
首页 > 资讯 > 精选 >ThreadLocal三大坑是什么
  • 940
分享到

ThreadLocal三大坑是什么

2023-06-15 12:06:16 940人浏览 薄情痞子
摘要

本篇内容主要讲解“ThreadLocal三大坑是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“ThreadLocal三大坑是什么”吧!内存泄露由于ThreadLocal的key是弱引用,因此

本篇内容主要讲解“ThreadLocal三大坑是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“ThreadLocal三大坑是什么”吧!

内存泄露

由于ThreadLocal的key是弱引用,因此如果使用后不调用remove清理的话会导致对应的value内存泄露。

@Test  public void testThreadLocalMemoryLeaks() {      ThreadLocal<List<Integer>> localCache = new ThreadLocal<>();     List<Integer> cacheInstance = new ArrayList<>(10000);      localCache.set(cacheInstance);      localCache = new ThreadLocal<>();  }

当localCache的值被重置之后cacheInstance被ThreadLocalMap中的value引用,无法被GC,但是其key对ThreadLocal实例的引用是一个弱引用,本来ThreadLocal的实例被localCache和ThreadLocalMap的key同时引用,但是当localCache的引用被重置之后,则ThreadLocal的实例只有ThreadLocalMap的key这样一个弱引用了,此时这个实例在GC的时候能够被清理。

ThreadLocal三大坑是什么

其实看过ThreadLocal源码的同学会知道,ThreadLocal本身对于key为null的Entity有自清理的过程,但是这个过程是依赖于后续对ThreadLocal的继续使用,假如上面的这段代码是处于一个秒杀场景下,会有一个瞬间的流量峰值,这个流量峰值也会将集群的内存打到高位(或者运气不好的话直接将集群内存打满导致故障),后面由于峰值流量已过,对ThreadLocal的调用也下降,会使得ThreadLocal的自清理能力下降,造成内存泄露。ThreadLocal的自清理是锦上添花,千万不要指望他雪中送碳。

相比于ThreadLocal中存储的value对象泄露,ThreadLocal用在WEB容器中时更需要注意其引起的ClassLoader泄露。

Tomcat官网对在web容器中使用ThreadLocal引起的内存泄露做了一个总结,详见:https://cwiki.apache.org/confluence/display/tomcat/MemoryLeakProtection,这里我们列举其中的一个例子。

熟悉Tomcat的同学知道,Tomcat中的web应用由Webapp Classloader这个类加载器的,并且Webapp Classloader是破坏双亲委派机制实现的,即所有的web应用先由Webapp classloader加载,这样的好处就是可以让同一个容器中的web应用以及依赖隔离。

下面我们看具体的内存泄露的例子:

public class MyCounter {   private int count = 0;   public void increment() {    count++;   }   public int getCount() {    return count;   }  }  public class MyThreadLocal extends ThreadLocal<MyCounter> {  }  public class LeakingServlet extends HttpServlet {   private static MyThreadLocal myThreadLocal = new MyThreadLocal();   protected void doGet(HttpServletRequest request,     HttpServletResponse response) throws ServletException, IOException {    MyCounter counter = myThreadLocal.get();    if (counter == null) {     counter = new MyCounter();     myThreadLocal.set(counter);    }    response.getWriter().println(      "The current thread served this servlet " + counter.getCount()        + " times");    counter.increment();   }  }

需要注意这个例子中的两个非常关键的点:

MyCounter以及MyThreadLocal必须放到web应用的路径中,保被Webapp Classloader加载

ThreadLocal类一定得是ThreadLocal的继承类,比如例子中的MyThreadLocal,因为ThreadLocal本来被Common Classloader加载,其生命周期与Tomcat容器一致。ThreadLocal的继承类包括比较常见的NamedThreadLocal,注意不要踩坑。

假如LeakingServlet所在的Web应用启动,MyThreadLocal类也会被Webapp Classloader加载,如果此时web应用下线,而线程的生命周期未结束(比如为LeakingServlet提供服务的线程是一个线程池中的线程),那会导致myThreadLocal的实例仍然被这个线程引用,而不能被GC,期初看来这个带来的问题也不大,因为myThreadLocal所引用的对象占用的内存空间不太多,问题在于myThreadLocal间接持有加载web应用的webapp classloader的引用(通过myThreadLocal.getClass().getClassLoader()可以引用到),而加载web应用的webapp classloader有持有它加载的所有类的引用,这就引起了Classloader泄露,它泄露的内存就非常可观了。

线程池中线程上下文丢失

ThreadLocal不能在父子线程中传递,因此最常见的做法是把父线程中的ThreadLocal值拷贝到子线程中,因此大家会经常看到类似下面的这段代码:

for(value in valueList){       Future<?> taskResult = threadPool.submit(new BizTask(ContextHolder.get()));//提交任务,并设置拷贝Context到子线程       results.add(taskResult);  }  for(result in results){      result.get();//阻塞等待任务执行完成  }

提交的任务定义长这样:

class BizTask<T> implements Callable<T>  {      private String session = null;         public BizTask(String session) {          this.session = session;      }          @Override      public T call(){          try {              ContextHolder.set(this.session);              // 执行业务逻辑          } catch(Exception e){              //log error          } finally {              ContextHolder.remove(); // 清理 ThreadLocal 的上下文,避免线程复用时context互串          }          return null;      }  }

对应的线程上下文管理类为:

class ContextHolder {      private static ThreadLocal<String> localThreadCache = new ThreadLocal<>();         public static void set(String cacheValue) {          localThreadCache.set(cacheValue);     }          public static String get() {          return localThreadCache.get();      }          public static void remove() {          localThreadCache.remove();      }     }

这么写倒也没有问题,我们再看看线程池的设置:

ThreadPoolExecutor executorPool = new ThreadPoolExecutor(20, 40, 30, TimeUnit.SECONDS, new LinkedBlockingQueue<Runnable>(40), new XXXThreadFactory(), ThreadPoolExecutor.CallerRunsPolicy);

其中最后一个参数控制着当线程池满时,该如何处理提交的任务,内置有4种策略

ThreadPoolExecutor.AbortPolicy //直接抛出异常  ThreadPoolExecutor.DiscardPolicy //丢弃当前任务  ThreadPoolExecutor.DiscardOldestPolicy //丢弃工作队列头部的任务  ThreadPoolExecutor.CallerRunsPolicy //转串行执行

可以看到,我们初始化线程池的时候指定如果线程池满,则新提交的任务转为串行执行,那我们之前的写法就会有问题了,串行执行的时候调用ContextHolder.remove();会将主线程的上下文也清理,即使后面线程池继续并行工作,传给子线程的上下文也已经是null了,而且这样的问题很难在预发测试的时候发现。

并行流中线程上下文丢失

如果ThreadLocal碰到并行流,也会有很多有意思的事情发生,比如有下面的代码:

class ParallelProcessor<T> {       public void process(List<T> dataList) {          // 先校验参数,篇幅限制先省略不写          dataList.parallelStream().forEach(entry -> {              doIt();          });     }      private void doIt() {          String session = ContextHolder.get();          // do something      }  }

这段代码很容易在线下测试的过程中发现不能按照预期工作,因为并行流底层的实现也是一个ForkJoin线程池,既然是线程池,那ContextHolder.get()可能取出来的就是一个null。我们顺着这个思路把代码再改一下:

class ParallelProcessor<T> {        private String session;         public ParallelProcessor(String session) {          this.session = session;      }          public void process(List<T> dataList) {          // 先校验参数,篇幅限制先省略不写          dataList.parallelStream().forEach(entry -> {              try {                  ContextHolder.set(session);                  // 业务处理                  doIt();              } catch (Exception e) {                  // log it              } finally {                  ContextHolder.remove();              }          });      }       private void doIt() {          String session = ContextHolder.get();          // do something      }  }

修改完后的这段代码可以工作吗?如果运气好,你会发现这样改又有问题,运气不好,这段代码在线下运行良好,这段代码就顺利上线了。不久你就会发现系统中会有一些其他很诡异的bug。原因在于并行流的设计比较特殊,父线程也有可能参与到并行流线程池的调度,那如果上面的process方法被父线程执行,那么父线程的上下文会被清理。导致后续拷贝到子线程的上下文都为null,同样产生丢失上下文的问题,关于并行流的实现可

到此,相信大家对“ThreadLocal三大坑是什么”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

--结束END--

本文标题: ThreadLocal三大坑是什么

本文链接: https://lsjlt.com/news/280221.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • ThreadLocal三大坑是什么
    本篇内容主要讲解“ThreadLocal三大坑是什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“ThreadLocal三大坑是什么”吧!内存泄露由于ThreadLocal的key是弱引用,因此...
    99+
    2023-06-15
  • 解决ThreadLocal获取不到值大坑
    目录1:问题起因2:问题复现3:分析问题4:如何解决1:问题起因 今天项目上测试环境,再给领导演示的时候出现了bug,很尴尬。于是我跟前端同学通过模拟请求,最后发现在调一个接口的时候...
    99+
    2023-05-19
    ThreadLocal获取不到值解决的 ThreadLocal坑
  • 什么是ThreadLocal
    这期内容当中小编将会给大家带来有关什么是ThreadLocal,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。一 引言ThreadLocal的官方API解释为:* ...
    99+
    2024-04-02
  • java中ThreadLocal是什么
    ThreadLocal是JDK包提供的,称为线程本地变量,它将变量与线程绑定在一起,为每一个线程维护一个独立的变量副本,通过ThreadLocal可以将对象的可见范围限制在同一个线程内,从而避免了线程安全问题,对解决多线程程序的并发问题有一...
    99+
    2024-04-02
  • java中的ThreadLocal是什么
    这篇文章将为大家详细讲解有关java中的ThreadLocal是什么,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。简单介绍ThreadLocal一般称为线程本地变量,它是一种特殊的线程绑定机制,将变量与线...
    99+
    2023-06-19
  • CSS三大特性是什么
    这篇文章主要介绍“CSS三大特性是什么”,在日常操作中,相信很多人在CSS三大特性是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”CSS三大特性是什么”的疑惑有所帮助!接...
    99+
    2024-04-02
  • Java三大框架是什么
    这篇文章给大家介绍Java三大框架是什么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java的优点是什么1. 简单,只需理解基本的概念,就可以编写适合于各种情况的应用程序;2. 面向对象;3. 分布性,Java是面向...
    99+
    2023-06-14
  • java三大特性是什么
    java三大特性是:1、面向对象,java最核心的特性之一,将现实世界中的事物抽象成类,并且用对象来描述和处理问题;2、平台无关性,java源代码经过编译后生成的是字节码,而不是机器码;3、高性能,通过即时编译和垃圾回收技术的应用,在运行时...
    99+
    2023-08-04
  • ThreadLocal使用方法是什么
    这篇“ThreadLocal使用方法是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“ThreadLocal使用方法是什么...
    99+
    2023-06-29
  • 前端三大框架是什么
    本篇内容介绍了“前端三大框架是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成! ...
    99+
    2024-04-02
  • 前端三大基础是什么
    本篇内容介绍了“前端三大基础是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成! ...
    99+
    2024-04-02
  • CSS的三大特性是什么
    这篇文章主要介绍“CSS的三大特性是什么”,在日常操作中,相信很多人在CSS的三大特性是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”CSS的三大特性是什么”的疑惑有所帮...
    99+
    2024-04-02
  • CSS的三大样式是什么
    这篇文章主要介绍“CSS的三大样式是什么”,在日常操作中,相信很多人在CSS的三大样式是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”CSS的三大样式是什么”的疑惑有所帮助!接下来,请跟着小编一起来学习吧...
    99+
    2023-06-26
  • React的三大属性是什么
    这篇文章主要为大家展示了“React的三大属性是什么”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“React的三大属性是什么”这篇文章吧。React三大属性props组件是封闭的,接收外部数据应...
    99+
    2023-06-29
  • w3c的三大标准是什么
    W3C(World Wide Web Consortium)是一个国际标准化组织,致力于推动和制定Web技术的标准。它的工作主要包括制定Web标准、推广Web技术和提供相关的指导和资源。W3C的三大标准是:1. HTML(Hypertext...
    99+
    2023-10-21
    w3c标准
  • java的三大体系是什么
    一、JavaSE 这是Java的标准版,它是Java编程的基础!可以在这款软件中完成基本的Java编程,学习和掌握Java的基本语法,以及面向对象的编程思想,不仅可以进行桌面应用的开发,也是学习JavaEE和JavaME的基础。 ...
    99+
    2023-10-29
    三大 体系 java
  • python中ThreadLocal的作用是什么
    这篇文章主要介绍“python中ThreadLocal的作用是什么”,在日常操作中,相信很多人在python中ThreadLocal的作用是什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”python中Th...
    99+
    2023-07-05
  • android开发三大框架是什么
    android开发三大框架是XUtil框架、volley框架、ImageLoader框架。详细介绍:1、XUtil框架有数据库模块、表明模块、网络模块、图片缓存模块等四大模块;2、volley框架的内容有JSON,图象等的异步下载、网络恳求...
    99+
    2023-08-14
  • java三大体系分别是什么
    java三大体系分别是Java SE、Java EE和Java ME。详细介绍:1、Java SE是Java平台的基础,用于开发桌面应用程序、嵌入式系统和移动设备等;2、Java EE是用于开发企业级应用程序的平台;3、Java ME是为嵌...
    99+
    2023-08-09
  • Spring中使用自定义ThreadLocal存储导致的坑及解决方法是什么
    本篇文章为大家展示了Spring中使用自定义ThreadLocal存储导致的坑及解决方法是什么,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Spring自定义ThreadLocal存储导致的坑Spr...
    99+
    2023-06-21
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作