您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > JAVA

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

时间:2023-09-18 15:15:04  来源:微信公众号  作者:闷骚的程序员

前几天有同学找我查一个空指针问题,JAVA 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。

现象

捕获异常打印日志的代码类似这样:

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

查到的日志是这样的:

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

异常堆栈丢了。

分析

在之前的一篇文章里已经验证过这种写法是可以正常打印异常和堆栈信息的:AI 自动补全的这句日志能正常打印吗?

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

 

https://www.oracle.com/java/technologies/javase/release-notes-introduction.html

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.

 

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

了解到这个信息后,翻了翻从服务上次发版以来的这条日志,果然最早的十几次打印是有异常堆栈的,后面就没有了。

解决方案

回溯历史日志,找到正常打印的堆栈信息,定位和解决问题;

也可以考虑在 JVM 参数里加上 -XX:-OmitStackTraceInFastThrow 参数,禁用优化;

本地复现

在本地写一个简单的程序复现一下:

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 参数时,运行结果如下:

...
count: 5783, stacktrace length: 2
count: 5784, stacktrace length: 2
count: 5785, stacktrace length: 0

Process finished with exit code 0

在我本机一般运行五六千次后,会出现异常堆栈丢失的情况。

添加 -XX:-OmitStackTraceInFastThrow 作为 JVM 参数时,运行结果如下:

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

运行了几百万次也不会出现异常堆栈丢失的情况,手动终止程序。

完整源码见:https://Github.com/mzlogin/java-notes/blob/master/src/org/mazhuang/StackTraceInFastThrowDemo.java



Tags:Java   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,不构成投资建议。投资者据此操作,风险自担。如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除。
▌相关推荐
前几天有同学找我查一个空指针问题,Java 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。现象捕获异常打印日志的代码类似这样:try { // ...} catch (Exception e...【详细内容】
2023-09-18  Tags: Java  点击:(0)  评论:(0)  加入收藏
在Java项目中,多个线程同时读写同一个文件可能会导致数据不一致的问题。这种情况下,当一个线程正在写入文件时,其他线程可能同时进行读取操作,导致读取到的数据是不一致或不完整...【详细内容】
2023-09-13  Tags: Java  点击:(6)  评论:(0)  加入收藏
在Java编程语言中,泛型(generics)是一种强劲的工具,它允许我们在类、接口和方法中使用类型参数。通过使用泛型,我们可以编写更加通用和可重用的代码。本文将深入探讨Java泛型,分析...【详细内容】
2023-09-13  Tags: Java  点击:(8)  评论:(0)  加入收藏
在此前我的文章中,曾分2篇详细探讨了下JAVA中Stream流的相关操作,2篇文章在掘金社区收获了累计 10w+阅读、2k+点赞以及 5k+收藏的记录。能够得到众多小伙伴的认可,是技术分享过...【详细内容】
2023-09-13  Tags: Java  点击:(1)  评论:(0)  加入收藏
译者 | 刘汪洋审校 | 重楼概括:这篇文章介绍了 JavaScript 中各种循环语句的特点和性能,以及如何根据不同的场景选择合适的循环方式。文章通过一些实例和测试,给出了一些使用循...【详细内容】
2023-09-13  Tags: Java  点击:(11)  评论:(0)  加入收藏
MySQL是一款常用的关系型数据库管理系统,为了保证数据的安全性和可靠性,备份与恢复策略是非常重要的。下面将介绍在Java实践中如何进行MySQL数据库的备份与恢复,并提供一些相关...【详细内容】
2023-09-12  Tags: Java  点击:(9)  评论:(0)  加入收藏
使用函数式编程可以减少代码重复,使代码更易于理解。Java编程语言以其面向对象的特性而闻名,但也因其冗长和繁琐的异常处理机制而而广受批评。当Java语言在1.8版本引入函数式...【详细内容】
2023-09-10  Tags: Java  点击:(3)  评论:(0)  加入收藏
随着互联网和大数据时代的到来,实时数据同步成为了许多企业面临的挑战。下面将介绍一种基于Change Data Capture(CDC)技术的解决方案,针对Java开发者在MySQL数据库中实现实时数...【详细内容】
2023-09-08  Tags: Java  点击:(15)  评论:(0)  加入收藏
在数据处理和分析过程中,数据去重是一个常见的需求。Java开发者可以使用MySQL数据库提供的丰富功能和优化技术来实现高效的数据去重。下面将介绍Java开发者如何利用MySQL数据...【详细内容】
2023-09-07  Tags: Java  点击:(13)  评论:(0)  加入收藏
根据 MDN:“闭包是捆绑在一起(封闭)的函数及其周围状态(词法环境)的引用的组合。换句话说,闭包使您可以从内部函数访问外部函数的作用域。在 JavaScript 中,每次创建函数时都会创建...【详细内容】
2023-09-07  Tags: Java  点击:(14)  评论:(0)  加入收藏
▌简易百科推荐
前几天有同学找我查一个空指针问题,Java 打印日志时,异常堆栈信息被吞了,导致定位不到出问题的地方。现象捕获异常打印日志的代码类似这样:try { // ...} catch (Exception e...【详细内容】
2023-09-18  闷骚的程序员  微信公众号  Tags:Java   点击:(0)  评论:(0)  加入收藏
在Java项目中,多个线程同时读写同一个文件可能会导致数据不一致的问题。这种情况下,当一个线程正在写入文件时,其他线程可能同时进行读取操作,导致读取到的数据是不一致或不完整...【详细内容】
2023-09-13  编程技术汇  今日头条  Tags:Java   点击:(6)  评论:(0)  加入收藏
在Java编程语言中,泛型(generics)是一种强劲的工具,它允许我们在类、接口和方法中使用类型参数。通过使用泛型,我们可以编写更加通用和可重用的代码。本文将深入探讨Java泛型,分析...【详细内容】
2023-09-13  佳慧慧    Tags:Java泛型   点击:(8)  评论:(0)  加入收藏
在此前我的文章中,曾分2篇详细探讨了下JAVA中Stream流的相关操作,2篇文章在掘金社区收获了累计 10w+阅读、2k+点赞以及 5k+收藏的记录。能够得到众多小伙伴的认可,是技术分享过...【详细内容】
2023-09-13  架构悟道    Tags:Java Stream   点击:(1)  评论:(0)  加入收藏
使用函数式编程可以减少代码重复,使代码更易于理解。Java编程语言以其面向对象的特性而闻名,但也因其冗长和繁琐的异常处理机制而而广受批评。当Java语言在1.8版本引入函数式...【详细内容】
2023-09-10  Java学研大本营    Tags:函数式编程   点击:(3)  评论:(0)  加入收藏
本文翻译自国外论坛 medium,原文地址:https://medium.com/@AlexanderObregon/maven-best-practices-tips-and-tricks-for-java-developers-438eca03f72bMaven 简介Maven 是一...【详细内容】
2023-09-09  waynblog  微信公众号  Tags:Maven   点击:(9)  评论:(0)  加入收藏
在数据处理和分析过程中,数据去重是一个常见的需求。Java开发者可以使用MySQL数据库提供的丰富功能和优化技术来实现高效的数据去重。下面将介绍Java开发者如何利用MySQL数据...【详细内容】
2023-09-07  编程技术汇  今日头条  Tags:MySQL   点击:(13)  评论:(0)  加入收藏
优雅处理Java与MySQL的并发访问冲突是开发分布式系统时需要考虑的重要问题。在多个线程或多个应用同时对MySQL数据库进行读写操作时,可能会出现数据一致性问题和性能问题。为...【详细内容】
2023-09-07  编程技术汇  今日头条  Tags:Java   点击:(13)  评论:(0)  加入收藏
Java Reflection 是一种强大的机制,允许开发人员在运行时动态地操作和扩展对象。它提供了许多功能,如获取类的信息、创建对象、调用方法、获取和设置字段的值等。在某些场景下...【详细内容】
2023-09-06  编程技术汇  今日头条  Tags:Java Reflection   点击:(25)  评论:(0)  加入收藏
来源 | OSCHINA 社区作者 | 抽刀断水-鹰影原文链接:https://my.oschina.net/u/3276866/blog/10091251背景在 java 开发中我们经常会遇到这样的场景,代码需要在服务器上(测试或...【详细内容】
2023-09-06  OSC开源社区    Tags:Java   点击:(14)  评论:(0)  加入收藏
站内最新
站内热门
站内头条