坚持这4点原则,让你更优雅的处理Java异常!

我们在日常编写代码的时候,不能保证程序不会出错,如何处理异常,是很考验一个人的编程功底的,曾经在面试过程中,我也遇到面试官问及对异常的处理问题。

虽然平时工作中对异常的处理已经习以为常了,但是并未认真梳理过,也没有考虑其中可能会出现的问题。

所以当时回答的并不是很好。这次专门梳理一下Java开发过程中如何更好的处理异常。

以便帮助大家在工作过程中,写出更加优雅的代码或者面试的时候能很好的阐述出来。

异常知识的复盘

在java中,Throwable类是一个专门用来处理异常的类。

我们开发过程中最常见的是它的两个子类,即Error 和Exception。

这两个实例对应着不同异常情况的处理机制,着重看一下他们的区别:

1、Exception是指程序在运行过程中,可以预料到的情况,进行捕获和处理后能够恢复程序运行的异常情况。

2、Error 是指程序运行中会存在的错误,这时程序中应该出现了比较严重问题,一般是不可恢复的,不需要捕获,也不便于捕获。比如OutOfMemoryError、NoClassDefFoundError等,它们在应用程序的控制和处理能力之外。

3、Exception还分为检查型异常和非检查型的异常。这两者有什么区别?

检查型的异常指,在编译期不强制捕获和处理的异常类型,如大名鼎鼎的NullPointerException、ArrayIndexOutOfBoundsException,他们大都属于RuntimeException,这一类异常通常能够通过编码来避免。

非检查型的异常指,在编译期必须捕获或抛出的异常类型,通常表示代码逻辑没有错误,但程序运行时可能会因为各种情况导致异常,如IOException,ConnectException等,这是在编写程序阶段是预料不到的。

异常处理机制

如何优雅的处理异常?

1、具体化原则

通常情况下,我们不要直接捕获Exception,应该捕获一个具体的异常(可以是自定义的),这样可以让代码逻辑更直观。

直接捕获Exception,可能捕获到其他不需要被捕获的异常,如RuntimeException,它应该直接被抛出,而不是进行捕获。

更不要捕获Throwable或者Error,因为这样你可以捕获到无法处理的错误。

最好的方式,就是直接捕获一个具体的异常,例如:读取文件的时候,直接捕获FileNotFoundException

try {
           FileInputStream inputStream = new FileInputStream("");
     }catch (FileNotFoundException e){
            
     }

2、不抛弃原则

try {
            FileInputStream inputStream = new FileInputStream("");
        }catch (FileNotFoundException e){
            e.printStackTrace();
        }

我见过大多程序员,在catch中处理异常都是像上面的代码一样,直接使用e.printStackTrace;

这样吞掉或者抛弃异常,会使我们无法知道哪里出了异常,也很难判断异常日志被输出到哪里去了,尤其是生产环境下。

那么,捕获到异常后究竟如何处理?

1)可以讲异常信息输出到现有的日志系统。

2)可以构建新的异常抛出,由上层业务处理。

比如,这里是一个文件加密的场景,在读取文件这一功能的时候,对于可能出现的FileNotFoundException,我们可以抛出一个自定义异常EncryptException,上层业务接受到该异常,可以展示相关信息给用户或者进行页面跳转等。

3、就近原则

就近原则,也可说是较早原则,就是我们在可能产生异常的代码段中尽可能较早的抛出异常。先看一段下面的代码:

public static void main(String[] args,String filename) {
        
        try {
            FileInputStream inputStream = new FileInputStream(filename);
        }catch (FileNotFoundException e){
        }
    }

这段代码中,filename参数可能存在NullPointerException,而FileInputStream本身也会存在异常的情况。像这种情况下,如果filename为null导致的错误,堆栈信息比较混乱,我们会很难查找到错误源,给排查工作带来很大困难。

对于,非检查型异常,前面也说过了,可以通过编码来避免或进行相应的处理。看下面的一段比较合理的代码:

public static void main(String[] args,String filename) {

        if(filename == null || filename.length() == 0){
            throw new NullPointerException("文件名/路径不可为空");
        }
        try {
            FileInputStream inputStream = new FileInputStream(filename);
        }catch (FileNotFoundException e){
        }
    }

这段代码中,对存在的异常,提前检查抛出,这时候程序运行出错,便能够很快定位到问题所在。

4、最小功能原则

我们代码基本上都坚持一个方法一个功能,也就是最小功能原则。

我们处理异常也是,通常异常代码块只处理异常代码的部分。

看下面一段代码。

public static void main(String[] args,String filename) {

        int aa = 6;
        try {
            int a = 50;
            int[] arr = new int[]{1,2};
            FileInputStream inputStream = new FileInputStream(filename);
        }catch (FileNotFoundException e){
            if(aa == 6){
                //
            }else{
                //
            }
        }
    }

这段代码中将一些不会产生异常的代码也放在了try-catch中,这样会产生额外的性能开销。

在catch中进行流程控制,这样的代码执行效率会更低。

所以,不要整段的代码都放在try-catch之中,也不要在异常代码中进行流程控制。

总结

1、处理好异常的前提,需要我们先理解 Throwable、Exception、Error 的关系以及常见的子类。

2、坚持具体化原则、不抛弃原则、就近原则、最小功能原则来处理异常。当然这四个原则是我根据处理异常时最重要的几点自己起的名字。

3、以后在面试的时候,就可以根据以上两点展开描述。工作中,按照四个原则也能很好的应对。

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

推荐阅读更多精彩内容