几种常见的JVM调优场景(建议收藏)

点击上方“芋道源码”,选择“设为星标”

管她前浪,还是后浪?

能浪的浪,才是好浪!

每天 10:33 更新文章,每天掉亿点点头发…

源码精品专栏

原创 | Java 2021 超神之路,很肝~中文详细注释的开源项目RPC 框架 Dubbo 源码解析网络应用框架 Netty 源码解析消息中间件 Roc++ketMQ 源码解析数据库中间件 Sharding-JDBC 和 MyCAT 源码解析作业调度中间件 Elastic-Job 源码解析分布式事务中间件 TCC-Transaction 源码解析Eureka 和 Hystrix 源码解析Java 并发源码

来源:cnblogs.com/spareyaya

/p/13174003.html

cpu占用过高死锁内存泄漏总结

最近很多小伙伴跟我说,自己学了不少JVM的调优知识,但是在实际工作中却不知道何时对JVM进行调优。今天,我就为大家介绍几种JVM调优的场景。

cpu占用过高

cpu占用过高要分情况讨论,是不是业务上在搞活动,突然有大批的流量进来,而且活动结束后cpu占用率就下降了,如果是这种情况其实可以不用太关心,因为请求越多,需要处理的线程数越多,这是正常的现象。话说回来,如果你的服务器配置本身就差,cpu也只有一个核心,这种情况,稍微多一点流量就真的能够把你的cpu资源耗尽,这时应该考虑先把配置提升吧。

第二种情况,cpu占用率长期过高 ,这种情况下可能是你的程序有那种循环次数超级多的代码,甚至是出现死循环了。排查步骤如下:

(1)用top命令查看cpu占用情况

几种常见的JVM调优场景(建议收藏)

这样就可以定位出cpu过高的进程。在linux下,top命令获得的进程号和jps工具获得的vmid是相同的:

几种常见的JVM调优场景(建议收藏)

(2)用top -Hp命令查看线程的情况

几种常见的JVM调优场景(建议收藏)

可以看到是线程id为7287这个线程一直在占用cpu

(3)把线程号转换为16进制

代码语言:javascript代码运行次数:0运行复制

[root@localhost ~]# printf "%x" 72871c77

记下这个16进制的数字,下面我们要用

(4)用jstack工具查看线程栈情况

代码语言:javascript代码运行次数:0运行复制

[root@localhost ~]# jstack 7268 | grep 1c77 -A 10"http-nio-8080-exec-2" #16 daemon prio=5 os_prio=0 tid=0x00007fb66ce81000 nid=0x1c77 runnable [0x00007fb639ab9000]   java.lang.Thread.State: RUNNABLE at com.spareyaya.jvm.service.EndlessLoopService.service(EndlessLoopService.java:19) at com.spareyaya.jvm.controller.JVMController.endlessLoop(JVMController.java:30) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) at java.lang.reflect.Method.invoke(Method.java:498) at org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:190) at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:138) at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:105)

通过jstack工具输出现在的线程栈,再通过grep命令结合上一步拿到的线程16进制的id定位到这个线程的运行情况,其中jstack后面的7268是第(1)步定位到的进程号,grep后面的是(2)、(3)步定位到的线程号。

从输出结果可以看到这个线程处于运行状态,在执行com.spareyaya.jvm.service.EndlessLoopService.service这个方法,代码行号是19行,这样就可以去到代码的19行,找到其所在的代码块,看看是不是处于循环中,这样就定位到了问题。

死锁

死锁并没有第一种场景那么明显,web应用肯定是多线程的程序,它服务于多个请求,程序发生死锁后,死锁的线程处于等待状态(WAITING或TIMED_WAITING),等待状态的线程不占用cpu,消耗的内存也很有限,而表现上可能是请求没法进行,最后超时了。在死锁情况不多的时候,这种情况不容易被发现。

可以使用jstack工具来查看

(1)jps查看java进程

代码语言:javascript代码运行次数:0运行复制

[root@localhost ~]# jps -l8737 sun.tools.jps.Jps8682 jvm-0.0.1-SNAPSHOT.jar

(2)jstack查看死锁问题

由于web应用往往会有很多工作线程,特别是在高并发的情况下线程数更多,于是这个命令的输出内容会十分多。jstack最大的好处就是会把产生死锁的信息(包含是什么线程产生的)输出到最后,所以我们只需要看最后的内容就行了

代码语言:javascript代码运行次数:0运行复制

Java stack information for the threads listed above:==================================================="Thread-4": at com.spareyaya.jvm.service.DeadLockService.service2(DeadLockService.java:35) - waiting to lock  (a java.lang.Object) - locked  (a java.lang.Object) at com.spareyaya.jvm.controller.JVMController.lambda$deadLock$1(JVMController.java:41) at com.spareyaya.jvm.controller.JVMController$$Lambda$457/1776922136.run(Unknown Source) at java.lang.Thread.run(Thread.java:748)"Thread-3": at com.spareyaya.jvm.service.DeadLockService.service1(DeadLockService.java:27) - waiting to lock  (a java.lang.Object) - locked  (a java.lang.Object) at com.spareyaya.jvm.controller.JVMController.lambda$deadLock$0(JVMController.java:37) at com.spareyaya.jvm.controller.JVMController$$Lambda$456/474286897.run(Unknown Source) at java.lang.Thread.run(Thread.java:748)Found 1 deadlock.

发现了一个死锁,原因也一目了然。

讯飞听见会议 讯飞听见会议

科大讯飞推出的AI智能会议系统

讯飞听见会议 19 查看详情 讯飞听见会议 内存泄漏

我们都知道,java和c++的最大区别是前者会自动收回不再使用的内存,后者需要程序员手动释放。在c++中,如果我们忘记释放内存就会发生内存泄漏。但是,不要以为jvm帮我们回收了内存就不会出现内存泄漏。

程序发生内存泄漏后,进程的可用内存会慢慢变少,最后的结果就是抛出OOM错误。发生OOM错误后可能会想到是内存不够大,于是把-Xmx参数调大,然后重启应用。这么做的结果就是,过了一段时间后,OOM依然会出现。最后无法再调大最大堆内存了,结果就是只能每隔一段时间重启一下应用。

内存泄漏的另一个可能的表现是请求的响应时间变长了。这是因为频繁发生的GC会暂停其它所有线程(Stop The World)造成的。

为了模拟这个场景,使用了以下的程序

代码语言:javascript代码运行次数:0运行复制

import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;public class Main {    public static void main(String[] args) {        Main main = new Main();        while (true) {            try {                Thread.sleep(1);            } catch (InterruptedException e) {                e.printStackTrace();            }            main.run();        }    }    private void run() {        ExecutorService executorService = Executors.newCachedThreadPool();        for (int i = 0; i  {                // do something...            });        }    }}

运行参数是-Xms20m -Xmx20m -XX:+PrintGC,把可用内存调小一点,并且在发生gc时输出信息,运行结果如下

代码语言:javascript代码运行次数:0运行复制

...[GC (Allocation Failure)  12776K->10840K(18432K), 0.0309510 secs][GC (Allocation Failure)  13400K->11520K(18432K), 0.0333385 secs][GC (Allocation Failure)  14080K->12168K(18432K), 0.0332409 secs][GC (Allocation Failure)  14728K->12832K(18432K), 0.0370435 secs][Full GC (Ergonomics)  12832K->12363K(18432K), 0.1942141 secs][Full GC (Ergonomics)  14923K->12951K(18432K), 0.1607221 secs][Full GC (Ergonomics)  15511K->13542K(18432K), 0.1956311 secs]...[Full GC (Ergonomics)  16382K->16381K(18432K), 0.1734902 secs][Full GC (Ergonomics)  16383K->16383K(18432K), 0.1922607 secs][Full GC (Ergonomics)  16383K->16383K(18432K), 0.1824278 secs][Full GC (Allocation Failure)  16383K->16383K(18432K), 0.1710382 secs][Full GC (Ergonomics)  16383K->16382K(18432K), 0.1829138 secs][Full GC (Ergonomics) Exception in thread "main"  16383K->16382K(18432K), 0.1406222 secs][Full GC (Allocation Failure)  16382K->16382K(18432K), 0.1392928 secs][Full GC (Ergonomics)  16383K->16382K(18432K), 0.1546243 secs][Full GC (Ergonomics)  16383K->16382K(18432K), 0.1755271 secs][Full GC (Ergonomics)  16383K->16382K(18432K), 0.1699080 secs][Full GC (Allocation Failure)  16382K->16382K(18432K), 0.1697982 secs][Full GC (Ergonomics)  16383K->16382K(18432K), 0.1851136 secs][Full GC (Allocation Failure)  16382K->16382K(18432K), 0.1655088 secs]java.lang.OutOfMemoryError: Java heap space

可以看到虽然一直在gc,占用的内存却越来越多,说明程序有的对象无法被回收。但是上面的程序对象都是定义在方法内的,属于局部变量,局部变量在方法运行结果后,所引用的对象在gc时应该被回收啊,但是这里明显没有。

为了找出到底是哪些对象没能被回收,我们加上运行参数-XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=heap.bin,意思是发生OOM时把堆内存信息dump出来。运行程序直至异常,于是得到heap.dump文件,然后我们借助eclipse的MAT插件来分析,如果没有安装需要先安装。

然后File->Open Heap Dump… ,然后选择刚才dump出来的文件,选择Leak Suspects

几种常见的JVM调优场景(建议收藏)

MAT会列出所有可能发生内存泄漏的对象

几种常见的JVM调优场景(建议收藏)

可以看到居然有21260个Thread对象,3386个ThreadPoolExecutor对象,如果你去看一下java.util.concurrent.ThreadPoolExecutor的源码,可以发现线程池为了复用线程,会不断地等待新的任务,线程也不会回收,需要调用其shutdown()方法才能让线程池执行完任务后停止。

其实线程池定义成局部变量,好的做法是设置成单例。

上面只是其中一种处理方法

在线上的应用,内存往往会设置得很大,这样发生OOM再把内存快照dump出来的文件就会很大,可能大到在本地的电脑中已经无法分析了(因为内存不足够打开这个dump文件)。这里介绍另一种处理办法:

(1)用jps定位到进程号

代码语言:javascript代码运行次数:0运行复制

C:UsersspareyayaIdeaProjectsmaven-projecttargetclassesorgexamplenet>jps -l24836 org.example.net.Main62520 org.jetbrains.jps.cmdline.Launcher129980 sun.tools.jps.Jps136028 org.jetbrains.jps.cmdline.Launcher

因为已经知道了是哪个应用发生了OOM,这样可以直接用jps找到进程号135988

(2)用jstat分析gc活动情况

jstat是一个统计java进程内存使用情况和gc活动的工具,参数可以有很多,可以通过jstat -help查看所有参数以及含义

代码语言:javascript代码运行次数:0运行复制

C:UsersspareyayaIdeaProjectsmaven-projecttargetclassesorgexamplenet>jstat -gcutil -t -h8 24836 1000Timestamp         S0     S1     E      O      M     CCS    YGC     YGCT    FGC    FGCT     GCT           29.1  32.81   0.00  23.48  85.92  92.84  84.13     14    0.339     0    0.000    0.339           30.1  32.81   0.00  78.12  85.92  92.84  84.13     14    0.339     0    0.000    0.339           31.1   0.00   0.00  22.70  91.74  92.72  83.71     15    0.389     1    0.233    0.622

上面是命令意思是输出gc的情况,输出时间,每8行输出一个行头信息,统计的进程号是24836,每1000毫秒输出一次信息。

输出信息是Timestamp是距离jvm启动的时间,S0、S1、E是新生代的两个Survivor和Eden,O是老年代区,M是Metaspace,CCS使用压缩比例,YGC和YGCT分别是新生代gc的次数和时间,FGC和FGCT分别是老年代gc的次数和时间,GCT是gc的总时间。虽然发生了gc,但是老年代内存占用率根本没下降,说明有的对象没法被回收(当然也不排除这些对象真的是有用)。

(3)用jmap工具dump出内存快照

jmap可以把指定java进程的内存快照dump出来,效果和第一种处理办法一样,不同的是它不用等OOM就可以做到,而且dump出来的快照也会小很多。

代码语言:javascript代码运行次数:0运行复制

jmap -dump:live,format=b,file=heap.bin 24836

这时会得到heap.bin的内存快照文件,然后就可以用eclipse来分析了。

总结

以上就是几种常见的JVM调优场景(建议收藏)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/298798.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月5日 00:00:29
下一篇 2025年11月5日 00:01:19

相关推荐

  • Polemos与Buff合作伙伴将数百万游戏玩家进入Web3生态系统

    正如6月23日polemos方法备受期待的代币生成事件(tge)日益临近,web3游戏基础设施平台宣布与buff建立一项具有里程碑意义的战略合作关系,后者是游戏行业中增长迅速的奖励平台之一。 这一合作标志着在传统Web 2.0游戏与去中心化的Web 3未来之间架起桥梁的重要一步。两大平台正携手努力,…

    2025年12月8日
    000
  • Memecoin市场中知识产权的模糊水域

    区块链领域中知识产权的灰色地带广为人知,尤其是在波动性极强的模因币市场。 最近一个典型案例是“松鼠花生”模因创作者马克·朗戈(Mark Longo)与知识产权保护之间的法律纠纷。 在加密世界里,IP权利常常成为雷区。虽然存在商业秘密、商标、技术专利以及内容版权等传统保护机制,但由于许多加密项目运作于…

    2025年12月8日
    000
  • 观看这个市场周期的顶级山寨币:不要错过Cardano,Blockdag,Vechain&Chainlink!

    探索2025年观看的顶级山寨币,包括blockdag,cardano,vechain和chainlink。了解为什么blockdag的2.98亿美元预售表现要优于当今市场上大多数加密硬币。 探索2025年观看的顶级山寨币,包括BlockDag,Cardano,Vechain和Chainlink。了解…

    2025年12月8日
    000
  • 欧易交易所APP官方正确地址

    获取欧易交易所APP官方正确地址需通过以下三个官方渠道:1.官方网站下载,访问官网域名并下载对应系统的版本;2.关注官方社交媒体账号获取最新下载信息;3.联系官方客服进行确认。同时,用户应警惕钓鱼网站、核对域名、安装杀毒软件、开启二次验证并避免泄露个人信息以保障账户安全。 欧易交易所APP官方正确地…

    2025年12月8日
    000
  • 比特币水龙头艺术装置纪念加密货币的兴起

    在这种情况下,基金投资者转型为数学艺术家的纳尔逊·塞尔斯(nelson saiers)创作了一件别具意义的雕塑,用以致敬比特币的崛起。 自2010年核心BTC贡献者加文·安德雷森(Gavin Andresen)推出了著名的比特币水龙头以来,比特币已经走过漫长的旅程。那时,加密货币诞生仅一年多时间,其…

    2025年12月8日
    000
  • 欧易交易所PC版 OKX交易所电脑版下载

    要下载并安装欧易交易所电脑版,请按照以下步骤操作:1. 访问欧易交易所官方网站,输入正确的官方网址并核实网站真实性;2. 找到首页或底部导航栏的“下载”入口;3. 选择“Windows”版本进行下载;4. 等待安装包下载完成;5. 双击安装包文件按提示完成安装;6. 启动客户端后登录现有账户或注册新…

    2025年12月8日
    000
  • SEC在两个加密ETF(位于dogecoin ETF和灰度HBAR HBAR ETF)上推迟了其决定。

    美国证券交易委员会(sec)再次推迟了对两款加密货币etf的审批决定,分别是bitwise dogecoin etf和grayscale hedera(hbar)etf。 根据Blockavise周一发布的报告,美国证券交易委员会已经扩大了对这两种加密ETF的审查范围,其中一款面向Dogecoin,…

    2025年12月8日
    000
  • Cardano(ADA)正在进入下一章

    加强网络的此举将使平台变得更加可扩展,可用和安全。 输入输出(IO)将在Cardano的下一章中向后拉窗帘,揭示了新的路线图和新的防组织措施。 作为使Cardano更具可扩展,可用和安全的努力的一部分,IO强调了如何加强网络的共识。 磨削攻击是一种恶意活动的一种形式,在这种活动中,一个坏演员与区块链…

    2025年12月8日
    000
  • Fomo的潮汐浪潮正在击中模因硬币现场,而Little Pepe($ lilpepe)正在高高骑行

    dogecoin(doge)正面临9600万个代币解锁的问题,这使得交易者纷纷转向一种名为 $ lilpepe 的模因币,该币具备第二层区块链功能。 FOMO热潮正在席卷模因币市场,Little Pepe($ Lilpepe)迅速崛起,在预售开启仅一天内就筹集了20万美元。当Dogecoin遭遇大量…

    2025年12月8日
    000
  • 系绳扩大了黄金策略,并在Elemental Altus版税中股份

    tether investments近日大举购入elemental altus特许权使用费公司的股份,进一步推进其以黄金和硬资产为支撑的金融基础设施战略。 加密巨头Tether通过此次投资,增持了以黄金特许权使用费为主营业务的Elemental Altus公司(ELE -0.36%),继续扩展其在硬…

    2025年12月8日
    000
  • 阻滞剂:2025年最佳趋势加密货币

    在预售阶段成功募集2.98亿美元,售出224亿枚bdag代币,该项目甚至在正式上线前便已积累强劲势头。 2025年6月正逐渐升温,加密交易者纷纷加码投入,这不仅仅是口头说说而已。Layer 1与DeFi领域的竞争愈发激烈,但只有少数项目展现出真正的增长动力、优质吸引力以及爆发潜力。 其中,Block…

    2025年12月8日
    000
  • 代币循环的重大增加继续与市场绩效保持权衡

    以下是你提供内容的伪原创版本,已保持原意不变,并保留图片位置: 该报告研究了五个加密项目:Taiko、Pyth、Morpho、Nyan 和 Spec,结果表明更高的流通量并未带来更强的需求增长。 根据代币主义者(前身为 Tokenunlocks)发布的 30 天供应分析显示,多个项目的代币循环供应量…

    2025年12月8日
    000
  • Maple Finance与Lido Finance合作伙伴提供由Steth支持的Stablecoin信用额度

    此举使机构在无需解锁以太坊(eth)的情况下借入稳定币,从而在释放流动性的同时继续享有以太坊的质押收益。 Maple Finance正与Lido Finance合作,推出由stETH支持的稳定币授信服务。stETH是代表已质押以太坊的流动性质押代币。 这项合作将使机构能够在不卖出ETH的前提下获取稳…

    2025年12月8日
    000
  • Unstaked的预售比赛朝1000万美元! Eth Eyes Surge&Sui面对供应问题

    随着主要加密货币市场走势不明朗,一个备受瞩目的预售项目正稳步前行。unstaked已成功筹集超过960万美元。 该项目名为Unstaked,旨在通过其创新的智能证明(POI)共识机制和人工智能技术重塑Web3生态系统,在其预售第20阶段已经募集到逾960万美元资金。 目前预售仍在继续,已有来自84,…

    2025年12月8日
    000
  • Mercurity Fintech Holding Inc.(NASDAQ:MFH)引入了一项主要战略计划,以比特币为中心

    在一项引人注目的战略举措中,mercurity fintech holding inc.(nasdaq:mfh)正式推出了一项聚焦比特币的核心发展计划。该公司致力于基于区块链技术的金融服务和产品开发,并宣布将启动一项高达80亿美元的融资方案,旨在建立长期持有的比特币储备金库,并将数字资产纳入其资本架…

    2025年12月8日
    000
  • 突破性技术解决方案使品牌能够在其生态系统中构建自定义信用卡产品,而无需将客户放置给传统银行

    coinbase(nasdaq:coin)作为美国领先的加密货币交易平台,成为首家采用cardless全新嵌入式信用卡平台构建专属卡产品的合作伙伴,推出了coinbase one卡。 在不断演进的金融科技领域,品牌正积极寻找创新方式以提升用户参与度、增强忠诚度并建立长期关系。过去,与传统银行合作发行…

    2025年12月8日
    000
  • AI助理平台Angel Twin将于6月18日推出其公用事业令牌$ ANGL

    公用事业代币$ angl的母公司angl inc.宣布其代币生成活动定于2025年6月18日启动 英属维尔京群岛,2025年6月12日,链闻 公用事业代币$ ANGL的母公司Angl Inc.正式公布,其代币生成事件(TGE)将于2025年6月18日举行,并将在UNISWAP上提供初始流动性。作为核…

    2025年12月8日
    000
  • 伊恩·莱尔(Ian Lyall)

    经验丰富的记者兼编辑伊恩·莱尔(ian lyall)在加入名为“积极主动”的执行编辑岗位时,带来了超过三十年的深厚行业积淀。 输出:一位英国媒体人近日就新闻机构采用人工智能撰写稿件的现象发表了看法。 他在提及自己所服务的这家专注于金融领域的新闻平台时表示,这里聚集了一批拥有丰富经验和专业素养的资深记…

    2025年12月8日
    000
  • CPSC召回的Sizhinai LED LED TUTU裙子由于电池摄入危险

    华盛顿特区 – 美国消费品安全委员会(CPSC)呼吁消费者立即停止使用Sizhinai品牌的LED Tutu连衣裙。 ![](data:image/jpeg;base64,/9j/4AAQSkZJRgABAQAAAQABAAD/2wCEAAkGBwgHBgkIBwgKCgkLDRYPDQ…

    2025年12月8日
    000
  • 比特币(BTC)的价格记录约1.85%

    比特币价格在过去24小时内下跌约1.85%,加密货币市场再次出现看跌情绪。 受美国生产者价格指数(PPI)超出预期的影响,加密货币市场整体走低。过去一天内,这一数据使比特币价格下跌了约1.85%。 目前,比特币的交易价格为107,249.8美元,交易量达到52034亿美元。 同时,山寨币市场也跟随比…

    2025年12月8日
    000

发表回复

登录后才能评论
关注微信