返回顶部
首页 > 资讯 > 后端开发 > Python >Java中ThreadLocal 导致内存 OOM 的原因分析
  • 722
分享到

Java中ThreadLocal 导致内存 OOM 的原因分析

2024-04-02 19:04:59 722人浏览 安东尼

Python 官方文档:入门教程 => 点击学习

摘要

目录原因分析正确的使用方式原因分析 ThreadLocal 导致内存 OOM 的原因是什么? ThreadLocal 底层通过 ThreadLocalMap 存储数据 源码如下:&n

原因分析

ThreadLocal 导致内存 OOM 的原因是什么?

ThreadLocal 底层通过 ThreadLocalMap 存储数据

源码如下: 

  • 当我们使用ThreadLocal.set()时,set的value与key(即业务自己定义的ThreadLocal类)会存储在ThreadLocalMap的Entry[]数组

源码如下:

  • 其中Entry是实现了一个弱引用WeakReference,Entry的key(即业务方定义的 ThreadLocal类)会被包装成一个弱引用当成Entry的key。Java的弱引用的定义是,当JVM执行垃圾回收扫描的时候,当发现只有弱引用的对象时,会立即回收此对象,这是ThreadLocal当初设计的时候防止内存溢出的一个手段

源码如下: 

虽然key被包装成了一个弱引用会被垃圾回收机制给回收,但是value在线程(Thread)不死亡时却可能存在一条强引用链.

由于 value是强引用,只要 Thread不死亡时,例如线程池,这条强引用链就会存在,那么value就不会回收,可能造成内存溢出

引用关系如下: Thread ==> ThreadLocalMap ==> Entry ==> value

但是这个消除强引用链的动作是需要业务方在get的情况下触发的,可能业务方并不会get、也可能get是key不为空,并不会触发 expungeStaleEntry 类。所以开发者要养成良好的习惯,记得用完 ThreadLocal 时,调一次ThreadLocal.remove()方法或者 ThreadLocal.set(null)

正确的使用方式

public class ThreadLocalTest {
    
    private static ThreadLocal<User> userThreadLocal = new ThreadLocal<>();
    public static void main(String[] args) throws InterruptedException {
        int threadNum = 10;
        List<Thread> threadList = Lists.newArrayList();

        for (int i = 0; i < threadNum; ++i) {
            long userId = i;
            Thread t = new Thread(() -> {
                try {
                    // 设置变量值
                    userThreadLocal.set(new User(userId, "lanxing" + userId, "2x"));
                    // 使用变量
                    doSomething();
                } finally {
                    // 移除变量
                    userThreadLocal.remove();   //移除ThreadLocal变量
                }
            }, "T" + i);
            threadList.add(t);
            t.start();
        }

        for (int i = 0; i < threadNum; ++i) {
            threadList.get(i).join();
        }
    }
    private static void doSomething() {
        log.info("Use ThreadLocal variable :{}", JSON.tojsONString(userThreadLocal.get()));
    }
}
@Data
@NoArgsConstructor
@AllArgsConstructor
class User {
    private Long id;
    private String name;
    private String level;
}

打印结果:

14:30:26.790 [T2] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":2,"level":"2x","name":"lanxing2"}
14:30:26.789 [T5] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":5,"level":"2x","name":"lanxing5"}
14:30:26.792 [T0] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":0,"level":"2x","name":"lanxing0"}
14:30:26.792 [T4] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":4,"level":"2x","name":"lanxing4"}
14:30:26.792 [T8] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":8,"level":"2x","name":"lanxing8"}
14:30:26.791 [T1] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":1,"level":"2x","name":"lanxing1"}
14:30:26.792 [T7] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":7,"level":"2x","name":"lanxing7"}
14:30:26.792 [T6] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":6,"level":"2x","name":"lanxing6"}
14:30:26.791 [T9] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":9,"level":"2x","name":"lanxing9"}
14:30:26.790 [T3] INFO io.zhengsh.order.tool.ThreadLocalTest - Use ThreadLocal variable :{"id":3,"level":"2x","name":"lanxing3"}

到此这篇关于Java中ThreadLocal 导致内存 OOM 的原因分析的文章就介绍到这了,更多相关Java 内存OOM 内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Java中ThreadLocal 导致内存 OOM 的原因分析

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

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

猜你喜欢
  • Java中ThreadLocal 导致内存 OOM 的原因分析
    目录原因分析正确的使用方式原因分析 ThreadLocal 导致内存 OOM 的原因是什么? ThreadLocal 底层通过 ThreadLocalMap 存储数据 源码如下:&n...
    99+
    2024-04-02
  • Java中ThreadLocal导致内存OOM的原因是什么
    本篇内容介绍了“Java中ThreadLocal导致内存OOM的原因是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!原因分析Thread...
    99+
    2023-06-30
  • ThreadLocal导致JVM内存泄漏原因探究
    目录为什么要使用ThreadLocal使用ThreadLocal具体实现引发内存泄漏的原因为什么要使用ThreadLocal 在一整个业务逻辑流程中,为了在不同的地方或者不同的方法中...
    99+
    2023-05-16
    JVM内存泄漏 JVM ThreadLocal内存泄漏
  • Java内存中出现OOM的原因有哪些
    Java内存中出现OOM的原因有哪些?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一,jvm内存区域1,程序计数器一块很小的内存空间,作用是当前线程所执行的字节码的行号指示器。...
    99+
    2023-05-31
    java oom ava
  • Android内存泄漏导致的原因有哪些
    这篇文章主要介绍“Android内存泄漏导致的原因有哪些”,在日常操作中,相信很多人在Android内存泄漏导致的原因有哪些问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Android内存泄漏导致的原因有哪些...
    99+
    2023-07-05
  • Java锁竞争导致sql慢日志原因分析
    线上在同步用户时,经常出现简单sql的慢日志。根据方法找到代码,发现方法内使用redisson进行锁操作,waiTime和leaseTime都为3秒,数据库操作比较简单,只是一个简单...
    99+
    2022-11-21
    sql在Java代码中执行很慢 Java sql慢日志
  • Android内存泄漏导致原因深入探究
    目录什么是内存泄露哪些操作会造成内存泄漏常见内存泄露问题1.资源性对象未关闭2.注册对象未注销3.非静态内部类的静态实例4.单例模式引起的内存泄露5.Handler临时性内存泄露6....
    99+
    2023-02-17
    Android内存泄漏原因 Android内存泄漏几种情况
  • Java中内存分配的原理分析
    Java中内存分配的原理分析,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。深入Java核心Java内存分配原理精讲Java内存分配与管理是Java的核心技术之一,之前我们曾...
    99+
    2023-06-17
  • Java中导致饥饿的原因是什么
    小编给大家分享一下Java中导致饥饿的原因是什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!如果一个线程因为CPU时间全部被其他线程抢走而得不到CPU运行时间,...
    99+
    2023-05-30
    java
  • @Async导致controller 404及失效原因解决分析
    目录前言排查过程排查结果分析1、接口4042、为何controller里面加了@Asyn异步就失效了总结前言 事情的起因是微服务A通过feign调用微服务B的某个接口,报了形如下的异...
    99+
    2024-04-02
  • 香港服务器内存不足是什么原因导致的
    导致香港服务器内存不足的原因:1、香港服务器应用程序池没有及时释放内存导致;2、香港服务器本身内存资源不足,无法满足当前业务需求导致;3、香港服务器上运行的程序过多,消耗内存资源过大导致;4、香港服务器正在遭受网络恶意攻击导致服务器资源被占...
    99+
    2024-04-02
  • 分析一些可能导致Golang速度变慢的原因
    随着云计算和大数据时代的到来,计算效率成为许多软件工程师面临的主要挑战之一。在这种情况下,Golang作为一种高效的编程语言,备受关注。然而,一些人声称Golang在处理某些情况下会变得异常缓慢。本文将分析一些可能导致Golang速度变慢的...
    99+
    2023-05-14
  • 内存泄露导致Android 中setVisibility() 失效原理
    目录一、前情概要二、摸索过程1、代码执行了吗?2、视图不显示的直接原因是什么?3、操作的视图是同一个吗?三、解决方案一、前情概要 目前,我在开发的一个 Android 项目需要各个功...
    99+
    2024-04-02
  • Java服务假死后续之内存溢出的原因分析
    目录一、现象分析二、原因排查三、故障解决一、现象分析 上篇博客说到,Java服务假死的原因是使用了Guava缓存,30分钟的有效期导致Full GC无法回收内存。经过优化后,已经不再...
    99+
    2024-04-02
  • PHP验证码复制粘贴导致不出现的原因分析
    标题:PHP验证码复制粘贴导致不出现的原因分析 在网站开发中,验证码是一种常见的安全验证机制,用于防止机器人恶意攻击和确保用户身份的安全。然而,有时候在页面中出现验证码不会显示的情况,...
    99+
    2024-02-29
    验证码 php 复制粘贴 验证码生成
  • java中ThreadLocal对象存储和获取的示例分析
    这篇文章主要介绍了java中ThreadLocal对象存储和获取的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。Java的特点有哪些Java的特点有哪些1.Java语...
    99+
    2023-06-14
  • Java中什么情况会导致内存泄漏
    这篇文章主要讲解了“Java中什么情况会导致内存泄漏”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java中什么情况会导致内存泄漏”吧!概念内存泄露:指程序中动态分配内存给一些临时对象,但对...
    99+
    2023-06-16
  • 使用高防服务器时导致内存不足的原因有哪些
    使用高防服务器时导致内存不足的原因:1.高防服务器自身的存储空间不足,内存配置无法满足现有的业务需求。2.运行的程序过多占用内存,也有可能是安装多余的软件。3.IIS应用程序池没有及时释放内存,没有根据网站自身情况进行设置自动回收时间。4....
    99+
    2024-04-02
  • Spring中使用自定义ThreadLocal存储导致的坑及解决
    目录Spring自定义ThreadLocal存储导致的坑一个容易想到的实现办法是使用ThreadLocalThreadlocal可能会产生内存泄露的问题及原理为什么会产生内存泄露?J...
    99+
    2024-04-02
  • java中ThreadLocal内存泄漏的解决方法
    小编给大家分享一下java中ThreadLocal内存泄漏的解决方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!java基本数据类型有哪些Java的基本数据类型...
    99+
    2023-06-14
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作