当前位置:首页 > 科技  > 软件

解决 Java 打印日志吞异常堆栈的问题

来源: 责编: 时间:2023-09-18 21:40:35 413观看
导读前几天有同学找我查一个空指针问题,Java 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。现象捕获异常打印日志的代码类似这样:try { // ...} catch (Exception e) { log.error("系统异常 customerCode:

前几天有同学找我查一个空指针问题,Java 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。sX628资讯网——每日最新资讯28at.com

sX628资讯网——每日最新资讯28at.com

现象

捕获异常打印日志的代码类似这样:sX628资讯网——每日最新资讯28at.com

try {    // ...} catch (Exception e) {    log.error("系统异常 customerCode:{},data:{}", customerCode, data, e);    // ...}

查到的日志是这样的:sX628资讯网——每日最新资讯28at.com

2023-06-26 11:11:11.111 ERROR 1 --- [pool-1-thread-1] c.mazhuang.service.impl.TestServiceImpl  : 系统异常 customerCode:123,data:{"name":"mazhuang","age":18}java.lang.NullPointerException: null

异常堆栈丢了。sX628资讯网——每日最新资讯28at.com

分析

在之前的一篇文章里已经验证过这种写法是可以正常打印异常和堆栈信息的:AI 自动补全的这句日志能正常打印吗?sX628资讯网——每日最新资讯28at.com

再三确认代码写法没问题,纳闷之下只好搜索了一下关键词「Java异常堆栈丢失」,发现了这篇文章:Java异常堆栈丢失的现象及解决方法,这里面提到的问题与我们遇到的一样,而且给出了 Oracle 官方文档里的相关说明:sX628资讯网——每日最新资讯28at.com

https://www.oracle.com/java/technologies/javase/release-notes-introduction.htmlsX628资讯网——每日最新资讯28at.com

The compiler in the server VM now provides correct stack backtraces for all “cold” built-in exceptions. For performance purposes, when such an exception is thrown a few times, the method may be recompiled. After recompilation, the compiler may choose a faster tactic using preallocated exceptions that do not provide a stack trace. To disable completely the use of preallocated exceptions, use this new flag: -XX:-OmitStackTraceInFastThrow.sX628资讯网——每日最新资讯28at.com

sX628资讯网——每日最新资讯28at.com

大致意思就是说,为了提高性能,JVM 会针对一些内建异常进行优化,在这些异常被某方法多次抛出时,JVM 可能会重编译该方法,这时候就可能会使用不提供堆栈信息的预分配异常。如果想要完全禁用预分配异常,可以使用 -XX:-OmitStackTraceInFastThrow 参数。sX628资讯网——每日最新资讯28at.com

了解到这个信息后,翻了翻从服务上次发版以来的这条日志,果然最早的十几次打印是有异常堆栈的,后面就没有了。sX628资讯网——每日最新资讯28at.com

解决方案

回溯历史日志,找到正常打印的堆栈信息,定位和解决问题;sX628资讯网——每日最新资讯28at.com

也可以考虑在 JVM 参数里加上 -XX:-OmitStackTraceInFastThrow 参数,禁用优化;sX628资讯网——每日最新资讯28at.com

本地复现

在本地写一个简单的程序复现一下:sX628资讯网——每日最新资讯28at.com

public class StackTraceInFastThrowDemo {    public static void main(String[] args) {        int count = 0;        boolean flag = true;        while (flag) {            try {                count++;                npeTest(null);            } catch (Exception e) {                int stackTraceLength = e.getStackTrace().length;                System.out.printf("count: %d, stacktrace length: %d%n", count, stackTraceLength);                if (stackTraceLength == 0) {                    flag = false;                }            }        }    }    public static void npeTest(Integer i) {        System.out.println(i.toString());    }}

不添加 -XX:-OmitStackTraceInFastThrow 作为 JVM 参数时,运行结果如下:sX628资讯网——每日最新资讯28at.com

...count: 5783, stacktrace length: 2count: 5784, stacktrace length: 2count: 5785, stacktrace length: 0Process finished with exit code 0

在我本机一般运行五六千次后,会出现异常堆栈丢失的情况。sX628资讯网——每日最新资讯28at.com

添加 -XX:-OmitStackTraceInFastThrow 作为 JVM 参数时,运行结果如下:sX628资讯网——每日最新资讯28at.com

...count: 3146938, stacktrace length: 2count: 3146939, stacktrace length: 2count: 3146940, stacktrace length: Process finished with exit code 137 (interrupted by signal 9: SIGKILL)

运行了几百万次也不会出现异常堆栈丢失的情况,手动终止程序。sX628资讯网——每日最新资讯28at.com

完整源码见:https://github.com/mzlogin/java-notes/blob/master/src/org/mazhuang/StackTraceInFastThrowDemo.javasX628资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-10425-0.html解决 Java 打印日志吞异常堆栈的问题

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: Python中30个常见的内置函数使用讲解(一)

下一篇: 【Python入门】为什么这门编程语言如此受欢迎?

标签:
  • 热门焦点
  • 对标苹果的灵动岛 华为带来实况窗功能

    继苹果的灵动岛之后,华为也在今天正式推出了“实况窗”功能。据今天鸿蒙OS 4.0的现场演示显示,华为的实况窗可以更高效的展现出实时通知,比如锁屏上就能看到外卖、打车、银行
  • 如何正确使用:Has和:Nth-Last-Child

    我们可以用CSS检查,以了解一组元素的数量是否小于或等于一个数字。例如,一个拥有三个或更多子项的grid。你可能会想,为什么需要这样做呢?在某些情况下,一个组件或一个布局可能会
  • 十个简单但很有用的Python装饰器

    装饰器(Decorators)是Python中一种强大而灵活的功能,用于修改或增强函数或类的行为。装饰器本质上是一个函数,它接受另一个函数或类作为参数,并返回一个新的函数或类。它们通常用
  • 三万字盘点 Spring 九大核心基础功能

    大家好,我是三友~~今天来跟大家聊一聊Spring的9大核心基础功能。话不多说,先上目录:图片友情提示,本文过长,建议收藏,嘿嘿嘿!一、资源管理资源管理是Spring的一个核心的基础功能,不
  • 一文掌握 Golang 模糊测试(Fuzz Testing)

    模糊测试(Fuzz Testing)模糊测试(Fuzz Testing)是通过向目标系统提供非预期的输入并监视异常结果来发现软件漏洞的方法。可以用来发现应用程序、操作系统和网络协议等中的漏洞或
  • 雅柏威士忌多款单品价格大跌,泥煤顶流也不香了?

    来源 | 烈酒商业观察编 | 肖海林今年以来,威士忌市场开始出现了降温迹象,越来越多不断暴涨的网红威士忌也开始悄然回归市场理性。近日,LVMH集团旗下苏格兰威士忌品牌雅柏(Ardbeg
  • 消费结构调整丨巨头低价博弈,拼多多还卷得动吗?

    来源:征探财经作者:陈香羽随着流量红利的退潮,电商的存量博弈越来越明显。曾经主攻中高端与品质的淘宝天猫、京东重拾“低价”口号。而过去与他们错位竞争的拼多多,靠
  • iQOO 11S屏幕细节公布:首发三星2K E6全感屏 安卓最好的直屏手机

    日前iQOO手机官方宣布,新一代电竞旗舰iQOO 11S将会在7月4日19:00正式与大家见面。随着发布时间的日益临近,官方关于该机的预热也更加密集,截至目前已
  • 英特尔Xe-HP项目终止,将专注Xe-HPC/HPG系列显卡

    据10 月 31 日消息报道,英特尔高级副总裁兼加速计算系统和图形事业部总经理 表示,Xe-HP“ Arctic Sound” 系列服务器 GPU 已经应用于 oneAPI devcloud 云服
Top