Java中的异常(上)

没有一名程序员希望自己在写程序的时候遇到异常,但是实际上异常是无法避免的,没有人能保证写出的程序不会出错,已无法保证用户会按照程序员的意愿来使用程序。既然异常无法避免,对异常的处理也就是必需的。异常处理已成为衡量一门语言是否成熟的标准之一,增加了异常处理机制后的程序有更好的容错性、更加健壮。目前的主流编程语言(如C++、C#、Python等)都提供了这种机制,Java也不例外。Java的异常机制主要依赖于trycatchfinallythrowsthrow这五个关键字。

最基本的异常处理——try...catch

下面是Java异常处理机制的语法结构:

try{
    //业务逻辑代码
    ...
}catch(Exception e){
    //异常处理代码
    ...
}

如果try块代码出现异常,系统会自动生成一个异常对象,该异常对象被提交给Java运行时环境(Runtime Environment),这个过程称为抛出(throw)异常。
当Java运行时环境接收到异常对象时,会自动寻找处理该异常对象的catch块。如果找到了合适的catch块,则把该异常对象交给改catch块处理,这个过程称为捕获(catch)异常;如果Java运行时环境无法找到捕获异常的catch块,则运行时环境终止,Java程序也随之退出。这就是Java中最基本的异常处理机制,即先抛出异常,再捕获异常(并处理)。
注意:try块与if语句不一样,即使try块里只有一条语句,花括号{}也不能省略。

异常类的继承体系

catch块都是专门用于处理异常类及其子类的异常实例。这句户乍一看很难理解,来看下面的例子:

异常的继承体系1

结合上图进行解释:当Java运行时环境接收到异常对象后,会一次判断该异常对象是否是catch块后异常类或其子类的实例,如果是,Java运行时环境将调用该catch块来处理该异常;否则将再次拿该异常对象和下一个catch块后的异常类进行比较,直至满足条件。
因此,try块后可以有多个catch块,这是为了针对不同的异常类提供不同的异常处理方式。当系统发生不同的意外情况时,系统会生成不同的异常对象,Java运行时环境就会根据该异常对象所属的异常类来决定使用哪个catch块来处理该异常。同时,通过在try块后提供多个catch块,也无须在一个catch块内使用ifswitch等语句判断异常类型进而采取不同的处理方式,使得异常处理逻辑更加细致、更有条理。
再来看一张图:

异常类的继承体系2

Java把所有的非正常情况分成两种:异常(Exception)和错误(Error)。Error错误,一般是指与虚拟机(JVM)相关的问题,如系统崩溃、虚拟机错误、动态连接失败等,这种错误无法恢复或不可能捕获,将导致应用程序中断。通常应用程序无法处理这些错误,因此应用程序不应该试图使用catch块来捕获Error对象。而对于异常,看个例子:

public class NullTest {
    public static void main(String[] args) {
        Date d = null;
        try {
            System.out.println(d.after(new Date()));
        } catch (NullPointerException e) {
            System.out.println("空指针异常!");
        } catch (Exception e) {
            System.out.println("未知异常!");
        }
    }
}

上面程序调用了一个null对象的after()方法,因此引发了NullPointerException。注意到程序中把对应Exception类的catch块放在最后,根据异常类的继承体系可知,这是因为如果把对应Exception类的catch块放在其他catch块的前面,出现异常后程序会直接进入该catch块,而排在它后面的catch块将永远不会被执行,这就违背了上文提到的对不同异常进行不同处理的原则。
实际上,进行异常捕获时不仅应该把Exception类对应的catch块放在最后,而且所有父类异常的catch块都应该排在子类异常catch块的后面(简称:先处理小异常,再处理大异常),否则会出现编译错误。将上面的代码稍作修改:

...
try {
    System.out.println(d.after(new Date()));
} catch (Exception e) {
    System.out.println("未知异常!");
} catch (NullPointerException e) {
    System.out.println("空指针异常!");
}

这时程序报错:

程序报错

错误信息是:这个异常已被Exception类的catch块捕获了。

访问异常信息

所有的异常对象都包含下面四个方法:

  • getMessage():返回该异常的详细描述字符串;
  • printStackTrace():将该异常的跟踪栈信息输出到标准错误输出;
  • printStackTrace(PrintStream s):将该异常的跟踪栈信息输出到指定输出流;
  • getStackTrace():返回该异常的跟踪栈信息。

还是刚才的例子:

...
try {
    System.out.println(d.after(new Date()));
} catch (NullPointerException e) {
    e.printStackTrace();
}

控制台输出为:

异常的跟踪栈信息

使用finally回收资源

有些时候,程序在try块里打开了一些物理资源(例如数据库连接、网络连接和磁盘文件等),这些物理资源都必须显示回收,因为Java的垃圾回收机制只回收堆内存中对象所占用的内存,不会回收任何物理资源。为解决这个问题,Java的异常处理机制提供了finally块。不管try块中的代码是否出现异常,也不管哪一个catch块被执行,甚者在try块或者catch块中执行了return语句,finally块总会被执行。因此,完整的Java异常处理语法结构为:

try{
    //业务逻辑代码
    ...
}catch(Exception e){
    //异常处理代码
    ...
}...
finally{
    //资源回收代码
}

注意:异常语法结构中只有try块是必需的;catch块和finally块都是可选的,但至少出现其中之一,也可以同时出现;finally块必须位于所有的catch块之后。
来看一个例子:

try {
    System.out.println(d.after(new Date()));
} catch (NullPointerException e) {
    e.printStackTrace();
    return;
} finally {
    System.out.println("finally块里的语句被执行了!");
}

控制台输出为:

控制台输出

程序的catch块有一条return语句。在通常情况下,一旦在方法里执行到return语句的地方,程序将立即结束该方法。但由控制台输出的结果可知,虽然return语句也强制方法结束,但一定会执行finally块里的语句。
作为对比,对上面的代码稍作修改:

try {
    System.out.println(d.after(new Date()));
} catch (NullPointerException e) {
    e.printStackTrace();
    System.exit(1);
} finally {
    System.out.println("finally块里的语句被执行了!");
}

再看控制台的输出:

控制台输出

程序中使用System.exit(1)语句来退出虚拟机,此时finally块将失去被执行的机会。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,493评论 18 399
  • 通俗编程——白话JAVA异常机制 - 代码之道,编程之法 - 博客频道 - CSDN.NEThttp://blog...
    葡萄喃喃呓语阅读 3,136评论 0 25
  • 初识异常(Exception) 比如我们在取数组里面的某个值得时候,经常会出现定义的取值范围超过了数组的大小,那么...
    iDaniel阅读 1,857评论 1 2
  • iOS10 从2017年1月1日起苹果不允许我们通过这个方法跳过ATS,也就是说强制我们用HTTPS,如果不这样的...
    moodi阅读 341评论 0 1
  • 一、前言 1、写了10多天的小程序代码,有兴趣的可以看我这篇小程序官方文档-小程序版【持续更新】,被坑得有点晕,突...
    gitKong阅读 2,053评论 14 30