NullPointerException异常没有异常栈打印问题追踪

摘要:
我下意识地查找异常堆栈,以查看导致空指针的代码行。然而,我发现日志中只打印了以下日志:nulljava.lang.NullPointerException:null。我想知道在打印日志时是否没有打印异常堆栈,所以我再次检查了源代码。记录器。错误这让我怀疑日志配置是否有问题。经过一番折腾,我发现这些配置都还可以。这个描述可以很好地解释我现在发现的问题。为了验证我的猜测,我在服务刚启动时查看了代码,发现异常堆栈正常打印。

今天去服务器后台看日志,发现有很多NullPointerException异常。我下意识的找异常栈,想看下到底是哪行代码导致了空指针。但是发现日志中只打印出了如下日志:

null
java.lang.NullPointerException: null

我怀疑是不是打印日志的时候是不是没有将异常栈打印出来,于是又去翻看源代码核实。但是发现日志打印也是正常的。

  logger.error(e.getMessage(),e);

这就纳闷了,于是有怀疑是不是日志配置有问题。折腾了一番发现这些配置都没问题。一时没有了思路,只好去求助万能的百度。

问题原因

我在网上找到了这么一段描述:

JVM虚拟机会对异常信息进行优化,当相同异常出现很多次,会认为它是热点异常,忽略掉异常堆栈信息;通过增加JVM参数:-XX:-OmitStackTraceInFastThrow可解决。

这个描述能很好的解释我现在发现的问题。代码中出现空指针异常的地方是一个定时任务在不停地调用,当这个异常出现次数太多时JVM就会将其过滤掉。

为了验证我的猜想,我去找了下这个服务刚刚启动时的代码,发现这个异常栈是正常打出的。验证了自己的猜想,通过异常栈也找到了导致空指针异常的代码。

问题重现

下面是自己写的一段代码来显示这个额问题:

public class NullPointStackMissBug {

    static Logger logger = LoggerFactory.getLogger(NullPointStackMissBug.class);

    public static void main(String[] args) {

        for (int i = 0; i < 100000 ; i++) {
            try{
                System.out.println("Loop:"+(i+1));
                String str = "test";
                if(true){
                    str = null;
                }
                str.toUpperCase();
            }catch (Exception e){
                logger.error(e.getMessage(),e);
            }
        }
    }

}

在JVM启动参数中增加:-XX:-OmitStackTraceInFastThrow后,异常就能正常输出。

免责声明:文章转载自《NullPointerException异常没有异常栈打印问题追踪》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇置换和轮换(新姿势,摘自黑书)QoS令牌桶工作原理下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

随便看看

influxdb简单使用

之前对influxdb有一个简单的了解和入门的使用,近期由于想使用influxdb做一点东西玩玩,又要捡起influxdb。而在influxdb下没有细分的表的概念,influxdb下的表在插入数据库的时候自动会创建。更多用户权限设置可以参看官方文档:https://docs.influxdata.com/influxdb/v1.0/query_langua...

Android开发 Camera2的CaptureRequest属性整理--完善中

当android.control.aeExposureCompensation改变时,即使AE锁定为ON,则相机设备将仍然调整其曝光值。如果android.control.aeMode是ON_ALWAYS_FLASH,现场可能会变得过度曝光。同样,AEprecapture触发取消了当AE已被锁定没有影响。见android.control.aeState为AE...

TCL基本语法2

TCL基本语法21、format和scan两个基本的函数,和C语言中的sprintf和scanf的作用基本相同。format将不同类型的数据压缩在字符串中,scan将字符串中的数据提取出来。setnameJacksetage100setworker[format"%sis%dyearsold"$name$age]puts$workerscan$worker"...

oracle 在sql中显示blob的字符串

最近在用oracle的过程中用到了对blob字段模糊查询的问题,对oracle来说,我并不是高手,找了很多的资料终于能够查出来了。以上只是自己做了个简单的处理,相信肯定有更好的方法,希望大家帮忙,但是感觉dbms_lob函数下的方法真的很好用。...

vue的富文本编辑器使用,并且添加显示当前输入字数

{模块:{工具栏:{标题:{script://indent〔{direction:text align:background:}.editor{line-height:}.ql editor{line-high:content:padding right:...

使用 supervisor 管理进程

Supervisor可以在Linux和Mac OS X上运行。Supervisor功能强大,提供了很多功能,但我们可能只需要使用其中的一小部分。为了方便起见,我们将配置分为两部分:管理程序和应用程序。首先,让我们看看supervisord的配置文件。...