iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >ThreadLocal在Tomcat中引起内存泄露怎么解决
  • 197
分享到

ThreadLocal在Tomcat中引起内存泄露怎么解决

2023-06-05 06:06:54 197人浏览 八月长安
摘要

这篇文章主要介绍“ThreadLocal在Tomcat中引起内存泄露怎么解决”,在日常操作中,相信很多人在ThreadLocal在Tomcat中引起内存泄露怎么解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答

这篇文章主要介绍“ThreadLocal在Tomcat中引起内存泄露怎么解决”,在日常操作中,相信很多人在ThreadLocal在Tomcat中引起内存泄露怎么解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”ThreadLocal在Tomcat中引起内存泄露怎么解决”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

定义

首先,我们要先谈一下定义,因为一堆人搞不懂内存溢出和内存泄露的区别。
内存溢出(OutOfMemory):你只有十块钱,我却找你要了一百块。对不起啊,我没有这么多钱。(给不起)
内存泄露(MemoryLeak):你有十块钱,我找你要一块。但是无耻的博主,不把钱还你了。(没退还)
关系:多次的内存泄露,会导致内存溢出。(博主不要脸的找你多要几次钱,你就没钱了,就是这个道理。)

危害

ok,大家在项目中有没遇到过java程序越来越卡的情况。
因为内存泄露,会导致频繁的Full GC,而Full GC 又会造成程序停顿,最后Crash了。因此,你会感觉到你的程序越来越卡,越来越卡,然后你就被产品经理鄙视了。顺便提一下,我们之所以JVM调优,就是为了减少Full GC的出现。
我记得,我曾经有一次,就遇到项目刚上线的时候好好的。结果随着时间的堆积,报了OutOfMemoryError: PermGen space
说到这个PermGen space,突然间,一阵洪荒之力,从博主体内喷涌而出,一定要介绍一下这个方法区,不过点到为止,毕竟这不是在讲《jvm从入门到放弃》。
方法区:出自java虚拟机规范, 可供各条线程共享运行时内存区域。它存储了每一个类的结构信息,例如运行时常量池(Runtime Constant Pool)、字段和方法数据、构造函数和普通方法的字节码内容。
上面讲的是规范,在不同虚拟机里头实现是不一样的,最典型的就是永久代(PermGen space)元空间(Metaspace)

jdk1.8以前:实现方法区的叫永久代。因为在很久远以前,java觉得类几乎是静态的,并且很少被卸载和回收,所以给了一个永久代的雅称。因此,如果你在项目中,发现堆和永久代一直在不断增长,没有下降趋势,回收的速度根本赶不上增长的速度,不用说了,这种情况基本可以确定是内存泄露。

jdk1.8以后:实现方法区的叫元空间。Java觉得对永久代进行调优是很困难的。永久代中的元数据可能会随着每一次Full GC发生而进行移动。并且为永久代设置空间大小也是很难确定的。因此,java决定将类的元数据分配在本地内存中,元空间的最大可分配空间就是系统可用内存空间。这样,我们就避开了设置永久代大小的问题。但是,这种情况下,一旦发生内存泄露,会占用你的大量本地内存。如果你发现,你的项目中本地内存占用率异常高。嗯,这就是内存泄露了。

如何排查

(1)通过jps查找java进程id。
(2)通过top -p [pid]发现内存占用达到了最大值
(3)jstat -gccause pid 20000 每隔20秒输出Full GC结果
(4)发现Full GC次数太多,基本就是内存泄露了。生成dump文件,借助工具分析是哪个对象太多了。基本能定位到问题在哪。

实例

在stackoverflow上,有一个问题,如下所示

I just had an interview, and I was asked to create a memory leak with Java. Needless to say I felt pretty dumb having no clue on how to even start creating one.

大致就是,因为面试需要手写一段内存泄露的程序,然后提问的人突然懵逼了,于是很多大佬纷纷给出回答。
案例一
此例子出自《算法》(第四版)一书,我简化了一下

    class stack{            Object data[1000];            int top = 0;            public void push(Object o){                    data[top++] = o;           }            public Object pop(Object o){             return data[--top];        }    }

当数据从栈里面弹出来之后,data数组还一直保留着指向元素的指针。那么就算你把栈pop空了,这些元素占的内存也不会被回收的。
解决方案就是

    public Object pop(Object o){         Object result = data[--top];        data[top] = null;        return result;    }

案例二
这个其实是一堆例子,这些例子造成内存泄露的原因都是类似的,就是不关闭流,具体的,可以是文件流,Socket流,数据库连接流,等等
具体如下,没关文件流

try {    BufferedReader br = new BufferedReader(new FileReader(inputFile));    ...    ...} catch (Exception e) {    e.printStacktrace();}

再比如,没关闭连接

try {    Connection conn = ConnectionFactory.getConnection();    ...    ...} catch (Exception e) {    e.printStacktrace();}

解决方案就是。。。嗯,大家应该都会。。你敢说你不会调close()方法。
案例三
讲这个例子前,大家对ThreadLocalTomcat中引起内存泄露有了解么。不过,我要说一下,这个泄露问题,和ThreadLocal本身关系不大,我看了一下官网给的例子,基本都是属于使用不当引起的。
在Tomcat的官网上,记录了这个问题。地址是:https://wiki.apache.org/tomcat/MemoryLeakProtection
不过,官网的这个例子,可能不好理解,我们略作改动。

public class HelloServlet extends HttpServlet{    private static final long serialVersionUID = 1L;    static class LocalVariable {        private Long[] a = new Long[1024 * 1024 * 100];    }    final static ThreadLocal<LocalVariable> localVariable = new ThreadLocal<LocalVariable>();    @Override    public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {        localVariable.set(new LocalVariable());    }}

再来看下conf下sever.xml配置

  <!--The connectors can use a shared executor, you can define one or more named thread pools-->    <Executor name="tomcatThreadPool" namePrefix="catalina-exec-"         maxThreads="150" minSpareThreads="4"/>

线程池最大线程为150个,最小线程为4个

Tomcat中Connector组件负责接受并处理请求,每来一个请求,就会去线程池中取一个线程。
在访问该servlet时,ThreadLocal变量里面被添加了new LocalVariable()实例,但是没有被remove,这样该变量就随着线程回到了线程池中。另外多次访问该servlet可能用的不是工作线程池里面的同一个线程,这会导致工作线程池里面多个线程都会存在内存泄露。

另外,servletdoGet方法里面创建new LocalVariable()的时候使用的是WEBappclassloader
那么
LocalVariable对象没有释放 -> LocalVariable.class没有释放 ->webappclassloader没有释放 -> webappclassloader加载的所有类也没有被释放,也造成了内存泄露。

除此之外,你在eclipse中,做一个reload操作,工作线程池里面的线程还是一直存在的,并且线程里面的threadLocal变量并没有被清理。而reload的时候,又会新构建一个webappclassloader,重复上述步骤。多reload几次,就内存溢出。
不过Tomcat7.0以后,你每做一次reload,会清理工作线程池中线程的threadLocals变量。因此,这个问题在tomcat7.0后,不会存在。

ps:ThreadLocal的使用在Tomcat的服务环境下要注意,并非每次web请求时候程序运行的ThreadLocal都是唯一的。ThreadLocal的什么生命周期不等于一次Request的生命周期。ThreadLocal与线程对象紧密绑定的,由于Tomcat使用了线程池,线程是可能存在复用情况。

到此,关于“ThreadLocal在Tomcat中引起内存泄露怎么解决”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: ThreadLocal在Tomcat中引起内存泄露怎么解决

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

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

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • ThreadLocal在Tomcat中引起内存泄露怎么解决
    这篇文章主要介绍“ThreadLocal在Tomcat中引起内存泄露怎么解决”,在日常操作中,相信很多人在ThreadLocal在Tomcat中引起内存泄露怎么解决问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答...
    99+
    2023-06-05
  • Android内存泄露怎么解决
    解决Android内存泄露问题的方法有以下几种:1. 避免长生命周期的引用:确保在不使用时及时释放对象的引用,如Activity中的...
    99+
    2023-09-29
    android
  • Java中的substring真的会引起内存泄露么
    这篇文章给大家介绍Java中的substring真的会引起内存泄露么,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。在Java中开发,String是我们开发程序可以说必须要使用的类型,String有一个substring...
    99+
    2023-06-17
  • 怎么在JVM中使用JFR解决内存泄露
    今天就跟大家聊聊有关怎么在JVM中使用JFR解决内存泄露,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。一个内存泄露的例子我们举一个内存泄露的例子,先定义一个大对象:public&nb...
    99+
    2023-06-15
  • qt程序内存泄露怎么解决
    解决Qt程序内存泄漏的方法如下:1. 使用对象的父子关系:在创建对象时,将对象的父对象设置为合适的父对象。这样,当父对象被销毁时,它...
    99+
    2023-08-18
    qt
  • android内存泄露怎么查看和解决
    Android内存泄露是指内存中的对象无法被及时释放,导致内存占用过高,影响应用的性能和稳定性。以下是查看和解决Android内存泄...
    99+
    2024-03-13
    android
  • Python内存泄露怎么查看和解决
    在Python中,内存泄露指的是由于对象在不再需要时没有被正确释放,导致内存占用不断增加的情况。下面是一些查找和解决Python内存...
    99+
    2023-10-22
    Python
  • java中ThreadLocal内存泄漏的解决方法
    小编给大家分享一下java中ThreadLocal内存泄漏的解决方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!java基本数据类型有哪些Java的基本数据类型...
    99+
    2023-06-14
  • Java 中出现内存泄露如何解决
    这期内容当中小编将会给大家带来有关Java 中出现内存泄露如何解决,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。首先,我用下面的命令监视进程:while ( sleep 1&...
    99+
    2023-06-17
  • 内存泄露导致Android中setVisibility()失效怎么解决
    本篇内容介绍了“内存泄露导致Android中setVisibility()失效怎么解决”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!一、前情...
    99+
    2023-07-02
  • Flex中出现内存泄露如何解决
    Flex中出现内存泄露如何解决,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Flex内存泄露举例:引用泄露:对子对象的引用,外部对本对象或子对象的引用都需要置null;系统类泄...
    99+
    2023-06-17
  • c++循环引用导致的内存泄露如何解决
    在 C++ 中,循环引用(circular reference)是指两个或多个对象相互引用,导致内存泄漏的情况。解决循环引用导致的内...
    99+
    2023-10-10
    c++
  • Java中的内存泄露问题和解决办法
    目录为什么会产生内存泄漏?内存泄漏对程序的影响?如何检查和分析内存泄漏?常见的内存泄漏及解决方法1、单例造成的内存泄漏2、非静态内部类创建静态实例造成的内存泄漏【已无】3、Handl...
    99+
    2024-04-02
  • 怎么理解MySQL中多源复制引起的内存泄漏
    怎么理解MySQL中多源复制引起的内存泄漏,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。场景 : MySQL-5.7, 所有的小版本(&l...
    99+
    2024-04-02
  • 怎么用JS理解IE的内存泄露
    这篇文章主要讲解了“怎么用JS理解IE的内存泄露”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么用JS理解IE的内存泄露”吧!一、前言IE6~8除了不遵守...
    99+
    2024-04-02
  • Java内存泄露的理解与解决是怎样的
    这篇文章给大家介绍Java内存泄露的理解与解决是怎样的,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java内存管理机制在C++ 语言中,如果需要动态分配一块内存,程序员需要负责这块内存的整个生命周期。从申请分配、到使...
    99+
    2023-06-17
  • Android中怎么利用Handler防止内存泄露
    今天就跟大家聊聊有关Android中怎么利用Handler防止内存泄露,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。 Handler可能导致的内存泄露及其优化 &...
    99+
    2023-05-30
    android handler
  • 怎么解析Flex内存泄露常见现象及解决方法
    怎么解析Flex内存泄露常见现象及解决方法,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Flex性能优化常用手法总结众所周知,目前国内的宽带应用并不是像很多发达...
    99+
    2023-06-17
  • Android中使用Handler造成的内存泄露如何解决
    这篇文章将为大家详细讲解有关Android中使用Handler造成的内存泄露如何解决,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。一、什么是内存泄露?  Java使用有向图机制,通过GC自动...
    99+
    2023-05-30
    android handler
  • Android中使用webview时出现内存泄露如何解决
    Android中使用webview时出现内存泄露如何解决,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。1.避免在xml直接写webview控件,这样会引用activity,...
    99+
    2023-05-31
    android webview
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作