异常分析初探

简介:

异常在Java中是非常重要的一个内容,了解异常有助于我们写出更加健壮的代码,本文将探讨一下几个问题:

1 异常是怎么被JVM捕获的?
2 新建异常实例是否耗时?为什么会耗时?是否能够避免?
3 为什么不推荐使用printStackTrace()打印异常信息?
4 spring jdbc运行时异常种类概要
5 什么时候应该抛出数据库运行时异常

1. 异常怎么被JVM捕获的

在了解这个之前首先介绍下java的异常表(Exception table),异常表是JVM处理异常的关键点,在java类中的每个方法中,会为所有的try-catch语句,生成一张异常表,存放在字节码的最后,该表记录了该方法内每个异常发生的起止指令和处理指令。
了解了异常表后,看下面这段java代码

public void catchException() {  
    long l = System.nanoTime();  
    for (int i = 0; i < testTimes; i++) { 
        try {  
            throw new Exception();  
        } catch (Exception e) { 
            //nothing to do
        }  
    }
    System.out.println("抛出并捕获异常:" + (System.nanoTime() - l));  
}

在这段try-catch结构的代码片段中,在try语句中抛出了一个异常并catch捕获该异常,对比下图该代码段的java字节码(使用javap -c命令)

下面请结合java代码和生成的字节码来看下面的指令分析:
0-4号: 执行try前面的语句
5号: 执行try语句前保存现场
6号: 执行try语句后跳转指令行,图中表示跳转到22
9-17号: try-catch代码生成指令,结合红色框图异常表,表示9-17号指令若有Exception异常抛出就执行17行指令.
16号: athrow 表示抛出异常
17号: astore 表示jvm将该异常实例存储到局部变量表中
22号: 恢复try语句执行前保存的现场
对比指令分析,再结合使用try-catch代码分析:
(1)若try没有抛出异常,则继续执行完try语句,跳过catch语句,此时就是从指令6跳转到指令22.
(2)若try语句抛出异常则执行指令17,将异常保存起来,若异常被方法抛出,调用方拿到异常可用于异常层次索引。

通过以上的分析,可以知道JVM是怎么捕获并处理异常,其实就是使用goto指令来做上下文切换,但是,异常真的像平时大家所认为的那样耗时吗?且看第二部分

2. 异常是否耗时?为什么会耗时?

想要知道异常是否耗时,最简单的方法就是来一段代码测试下,如下代码片段

public class ExceptionTest {

private int counts;  
  
public ExceptionTest(int counts) {  
    this.counts = counts;  
}  

public void newObject() {  
    long l = System.nanoTime();  
    for (int i = 0; i < counts; i++) {  
        new Object();  
    }  
    System.out.println("建立基础对象:" + (System.nanoTime() - l));  
}  

public void newOverridObj() {  
    long l = System.nanoTime();  
    for (int i = 0; i < counts; i++) {  
        new Child();  
    }  
    System.out.println("建立继承对象:" + (System.nanoTime() - l));  
} 

public void newException() {  
    long l = System.nanoTime();  
    for (int i = 0; i < counts; i++) {  
        new Exception();
    }  
    System.out.println("新建异常对象:" + (System.nanoTime() - l));  
}  

public void catchException() {  
    long l = System.nanoTime(); 
    for (int i = 0; i < counts; i++) { 
        try {  
            throw new Exception();
        } catch (Exception e) { 
            //nothing to do
        }  
    }
    System.out.println("抛出并捕获异常:" + (System.nanoTime() - l));  
} 

public static void main(String[] args) {  
    ExceptionTest test = new ExceptionTest(10000);  
    test.newObject();  
    test.newOverridObj();
    test.newException();  
    test.catchException();  
}

}

在运行1W次的耗时输出为(单位:纳秒):
建立基础对象: 2181164
建立继承对象: 4920114
新建异常对象: 22716147
抛出并捕获异常: 101761933

很清晰的可以看到新建一个异常对象比新建一个普通对象在耗时上多一个数量级,抛出并捕获异常的耗时比新建一个异常在耗时上也要多一个数量级。

答案已经很清晰明了,创建一个异常对象却是要比一个普通对象耗时多,捕获一个异常耗时更甚。
为什么新建一个异常对象这么耗时?且看源码:
在java中,所有的异常都继承自Throwable类,Throwable的构造函数

        fillInStackTrace();
    }```
有个nativ方法`public synchronized native Throwable fillInStackTrace();`这个方法会存入当前线程的堆栈信息。也就是说每次创建一个异常实例都会把堆栈信息存一遍。这就是时间开销的主要来源了。

这个时候我们可以下一个结论:**新建异常对象比创建一个普通对象是要更加的耗时。**

能避开创建异常的这个耗时吗?答案是可以的,如果在程序中我们不关心异常抛出的异常占信息,我们可以自己定义一个异常继承自已有的异常类型,并写一个方法覆盖掉fillInStackTrace方法就行了。

### 3. 使用printStackTrace()打印异常信息分析 ###
还是通过源码来分析:

public void printStackTrace() {

printStackTrace(System.err);

}

public void printStackTrace(PrintStream s) {

    synchronized (s) {
        s.println(this);
        StackTraceElement[] trace = getOurStackTrace();
        //for循环stack trace,at……信息是不是很熟悉
        for (int i=0; i < trace.length; i++)
            s.println("\tat " + trace[i]);

        Throwable ourCause = getCause();
        if (ourCause != null)
            ourCause.printStackTraceAsCause(s, trace);
    }

}

//递归函数,打印cause信息
private void printStackTraceAsCause(PrintStream s,

                                    StackTraceElement[] causedTrace)
{
    // assert Thread.holdsLock(s);

    // Compute number of frames in common between this and caused
    StackTraceElement[] trace = getOurStackTrace();
    int m = trace.length-1, n = causedTrace.length-1;
    while (m >= 0 && n >=0 && trace[m].equals(causedTrace[n])) {
        m--; n--;
    }
    int framesInCommon = trace.length - 1 - m;
    
    //熟悉的输出格式
    s.println("Caused by: " + this);
    for (int i=0; i <= m; i++)
        s.println("\tat " + trace[i]);
    if (framesInCommon != 0)
        s.println("\t... " + framesInCommon + " more");

    // Recurse if we have a cause
    Throwable ourCause = getCause();
    //递归
    if (ourCause != null)
        ourCause.printStackTraceAsCause(s, trace);
}
在这段代码中可以看到在调用printStackTrace()方法的过程中会去爬栈,包括一个for循环和一个递归调用,当异常抛出的层次比较深的时候这个是非常耗时。

了解了异常捕获和开销,下面看看使用spring jdbc运行时主要的异常种类

### 4. spring jdbc运行时异常种类概要 ###
先上一张图![](http://img4.tbcdn.cn/L1/461/1/90be9a49d90191d894b13f9b248ac6866d229c43)
数据库运行时异常的父类是DataAccessException,这个类包括三个子类,由于在我们的工程项目中是使用tddl直接对数据连接,我们一般只关系数据库对数据的操作(select、insert、update、delete),所以图中第二种和第三种异常本节不做讨论,主要看看第一种异常的种类与发送的情形,下面列出了   
-     CleanupFailureDataAccessException 
      数据库操作结束后的清理工作出现异常(关闭连接等)
-     DataIntegrityViolationException
      尝试插入或更新数据后,约束问题上的一些抛出(比如重复插入唯一键值)
-     DataRetrievalFailureException,
       数据检索查找时候抛出一些异常(比如通过一个标识查找数据引发IO等问题)
-     DataSourceLookupFailureException
      指定的数据源无法得到抛出异常
-     InvalidDataAccessApiUsageException
       使用了不正确的数据库API抛出异常()
-     InvalidDataAccessResourceUsageException,
       使用数据库资源不正确抛出异常,例如使用错误的SQL
-     NonTransientDataAccessResourceException,
       资源访问永久失败会抛出异常
-     PermissionDeniedDataAccessException,
       没有权限访问数据的某些表字段会抛出异常
-     UncategorizedDataAccessException
       其他的一些不能使用更准确的描述的异常

### 5. 什么时候应该抛出数据库运行时异常 ###
本节主要是结合项目代码来讲,项目代码一般都分成三个部分:DAO、Service、Controller。其中DAO用来做数据库的基本操作(select、insert、update、delete),Service一般用来处理具体的业务逻辑、Controller来做http请求控制。

在代码中的DAO层面一般我们都没有对异常进行抛出,导致我们在Service层和Controller也没有关注过数据库操作的异常,最终异常都是被JVM捕获并输出,这样会带来一个不好的结果,就是我们在自己的代码中并没有掌控住数据库操作这部分代码的执行细节。

比如说,一张表用来存储店铺的动态,店铺ID和动态的发布时间为唯一主键。考虑到并发性,如果一个店铺在同一个时间来了2条动态,那么就会有一条动态会插入失败。而因为我们没有在DAO层去捕获唯一键插入失败异常,会使得在业务处理层Service没有关注到这种异常的产生。最终使得我们会忽略掉某其中的某一条动态,且不知道忽略的到底是哪一条!

有了上面这个例子,可以知道在insert和update语句上,如果唯一主键有可能在并发的情况下产生不可预料的结果那么我们应该在DAO层就抛出一个唯一主键冲突异常。这样在写业务逻辑的时候会需要我们必须捕获这个异常并做异常逻辑处理。

总结下,在DAO层,对于数据库操作可能引起的需要Service层来处理的异常,在DAO层的方法定义上必须显示的抛出异常。


参考资料:
http://icyfenix.iteye.com/blog/857722
http://itindex.net/detail/46560-spring-jdbc
http://blog.csdn.net/p106786860/article/details/11795771
http://www.cnblogs.com/chenssy/p/3438130.html
http://happyenjoylife.iteye.com/blog/1061495
http://www.zhihu.com/question/21405047
http://www.blogjava.net/liudecai/archive/2009/04/08/264460.html
http://itindex.net/detail/47791-exception-%E6%80%A7%E8%83%BD-%E9%97%AE%E9%A2%98
目录
相关文章
|
3天前
|
SQL 运维 算法
链路诊断最佳实践:1 分钟定位错慢根因
本文聚焦于线上应用的风险管理,特别是针对“错”(程序运行不符合预期)和“慢”(性能低下或响应迟缓)两大类问题,提出了一个系统化的根因诊断方案。
|
8月前
|
运维 供应链 监控
根因分析
根因分析
266 0
|
SQL 缓存 运维
使用篇丨链路追踪(Tracing)很简单:链路实时分析、监控与告警
使用篇丨链路追踪(Tracing)很简单:链路实时分析、监控与告警
6584 12
使用篇丨链路追踪(Tracing)很简单:链路实时分析、监控与告警
|
存储 SQL 监控
SLS新版告警自助排查系列之告警监控
在SLS告警中,告警监控通过对数据源的查询监控,然后产生告警,并将告警发送到告警管理,告警管理会对告警进行降噪处理包括合并抑制静默后,在将告警发送给行动管理,最终发送通知到用户配置的接收渠道。在整个过程中,告警监控作为告警的源头,决定着告警是否能准确的发出。在配置告警监控规则时,配置不当或者配置错误都会导致告警不能触发或者不是希望的触发。本文主要介绍在告警监控中如何进行自助排查问题。
629 0
|
SQL JSON 运维
如何使用下探分析定位多维指标异常根因
在系统运维过程中,关键指标的异常变化往往意味着服务异常、系统故障等等。因此我们往往会对一些关键指标进行自动巡检,例如异常检测和时序预测等等,及时感知指标的异常变化,了解系统的健康状况。对于复杂系统来说,感知到异常后直接在系统层面根因定位可能是十分困难的。因此我们需要一些手段缩小问题的排查范围或者直接定位问题,如使用 trace 根因分析等等。阿里云日志服务上线了下探分析功能,用于多维指标异常根因定位。我们将介绍该功能的使用场景和使用案例。
800 0
如何使用下探分析定位多维指标异常根因
|
8月前
|
SQL 缓存 监控
链路追踪(Tracing)其实很简单——链路实时分析、监控与告警
作者:夏明(涯海) 创作日期:2022-07-17 专栏地址:【稳定大于一切】【稳定大于一切】前面两小节我们介绍了单链路的筛选与轨迹回溯,是从单次请求的视角来分析问题,类似查询某个快递订单的物流轨迹。但是,单次请求无法直观的反映应用或接口的整体服务状态,经常会由于网络抖动、宿主机 GC 等原因出现偶...
309 0
链路追踪(Tracing)其实很简单——链路实时分析、监控与告警
|
8月前
|
SQL 运维 前端开发
链路追踪(Tracing)其实很简单——多维链路筛选
作者:夏明(涯海) 创作日期:2022-07-14 专栏地址:【稳定大于一切】【稳定大于一切】上一小节我们介绍了如何通过调用链和关联信息进行问题诊断,但是,细心的读者可能会有一个疑问,整个系统有那么多的调用链,我怎么知道哪条链路才是真正描述我在排查的这个问题?如果找到了不相符的链路岂不是会南辕北辙?...
356 0
链路追踪(Tracing)其实很简单——多维链路筛选
|
存储 缓存 运维
如何实现全链路系统问题90%精准诊断?
DevKit系统诊断工具是鲲鹏性能分析工具的子工具之一,能够针对内存、网络、存储等常见故障和异常,提供精准定位和诊断能力,帮助用户识别出源代码中的问题点,提升程序的可靠性,故障定位准确率高达90%。
246 0
如何实现全链路系统问题90%精准诊断?
|
消息中间件 监控 NoSQL
|
存储 资源调度 运维
如何通过链路追踪进行定时任务诊断
分布式任务调度平台 SchedulerX 有效地将用于微服务场景下的可视化全链路追踪能力引入至定时任务处理场景,这将大大提升定时任务在运行时可观测能力,有效地帮助定时任务执行过程中异常、耗时、执行卡住等问题的定位分析。
如何通过链路追踪进行定时任务诊断