《Effective Java》——异常

只针对异常情况才使用异常

异常设计的初衷就是针对程序的不正常情形所使用的,不要使用异常来控制程序的执行流程

对可恢复的情况使用受检异常,对编程错误使用非受检异常

Java设计了三种可以抛出的结构

  • checked exception
  • runtim exception
  • error

error一般情况下,约定俗称有虚拟机使用,表示资源不足等错误

对可恢复的情况使用受检异常,对编程错误使用非受检异常

避免不必要的使用checked exception

使用checked exception会给客户端程序员带来很大的负担,只有满足如下两个条件,才应该使用checked exception

  • 即使正确调用该api此种情况也可能发生
  • 对于异常的发生,程序员可以立即采取有用的动作

否则,应该使用runtime exception

优先使用标准的异常

重用现有的异常,有三个好处

  • 使你的api易于学习和使用
  • 对于使用api的程序来说,可读性会更好
  • 异常类越少,装载时的开销也越小

常用的异常有如下几个

  • IllegalArgumentException
  • IllegalStateException
  • NullPointerException
  • IndexOutOfBoundException
  • ConcurrentModificationException
  • UnsupportedOperationException

抛出与抽象相对应的异常

当方法要传递底层实现抛出的异常时,要进行异常转译,不要使底层实现的异常污染了上层api

例如

catch(IOException e){
    throw new ServiceException(e);
}

通常有两种转译方式,一种是直接异常的转译,一种是异常链的转译

上面的例子就是异常链的转译,调用方可以通过异常获得更底层的异常,方便排查问题,直接转译就是不传递异常链,例如

catch(IOException e){
    throw new ServiceException("io exception");
}

以上只是针对一定要传递底层异常情况下的处理方法,对于底层的方法,首选方案是处理掉底层异常,不将异常的影响传递到上层去,一般有两条途径来实现

  • 对参数进行检查,避免调用底层方法发现异常
  • 绕开异常,将异常记录下来,不传播出去

每个方法抛出的异常都要有文档

为checked exception和unchecked exception都建立文档说明,但是不要在方法声明中标记unchecked exception

在细节消息中包含能捕获失败的信息

要记录下产生该异常的各种原因,最好能使异常自身记录下产生问题的原因,例如在异常的构造函数中包含一些特别的字段,就像在IndexOutOfBoundsException中可以有如下构造方法

public IndexOutOfBoundsException(int lowerBound, int upperBound, int index){
    ...

努力使失败保持原子性

失败的方法调用应该使对象保持在被调用之前的状态,有几种途径可以实现这个目的

  • 使用不可变对象
  • 在改变状态之前检查参数有效性,提前抛出异常
  • 将可能失败的操作放在改变状态之前执行,这样失败是可以提前结束,不至于使对象处于不一致的状态
  • 写恢复代码(不常用),一般是对于已经持久化的数据
  • 使操作在一个临时拷贝上执行

有时,并不总能实现失败的原子性,例如两个线程不安全的并发修改对象的状态

规范上来讲,任何调用失败都应该使对应状态位于之前的状态,如何有不一致,应该在文档中描述出来

不要忽略异常

对于异常,除了关闭InputStream时,其他时候都不要忽略异常

对异常

  • 要么处理
  • 要么传播出去

此建议对于checked 和unchecked异常都使用

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-11-29 05:55:28

《Effective Java》——异常的相关文章

使用 IDEA 创建 Maven Web 项目 (异常)- Disconnected from the target VM, address: '127.0.0.1:59770', transport: 'socket'

运行环境: JDK 版本:1.8 Maven 版本:apache-maven-3.3.3 IDEA 版本:14 maven-jetty-plugin 配置: <plugin> <groupId>org.eclipse.jetty</groupId> <artifactId>jetty-maven-plugin</artifactId> <configuration> <webAppSourceDirectory>${pro

mybaits非配置原因,导致SqlSession was not registered for synchronization异常

今天运行程序时报了 SqlSession [[email protected]] was not registered for synchronization because synchronization is not active [11:03:17]-Closing non transactional SqlSession [[email protected]] 由于异常是集中处理的,所以报了这样的错误,查了半天,网上结果都是说配置文件出错的,可是我的项目配置文件肯定是没错的,因为项目都开

异常笔记--java编程思想

开一个新的系列,主要记一些琐碎的重要的知识点,把书读薄才是目的...特点: 代码少,概念多... 1. 基本概念 异常是在当前环境下无法获得必要的信息来解决这个问题,所以就需要从当前环境跳出,就是抛出异常.抛出异常后发生的几件事: 1.在堆上创建异常对象. 2.当前的执行路径中止                                          3. 当前环境抛出异常对象的引用.                                         4. 异常处理机制接

爱上MVC~业务层刻意抛出异常,全局异常的捕获它并按格式返回

对于业务层的程序的致命错误,我们一直的做法就是直接抛出指定的异常,让程序去终断,这种做法是对的,因为如果一个业务出现了致命的阻塞的问题,就没有必要再向上一层一层的返回了,但这时有个问题,直接抛异常,意味着服务器直接500了,前端如何去显示,或者如果你是API的服务,如果为前端返回,如果是500,那直接就挂了,哈哈! 下面是在MVC环境下优化的全局异常捕获代码(非API) /// <summary> /// 全局异常捕获 /// </summary> public class Glo

Laravel 5.4 中的异常处理器和HTTP异常处理实例教程

错误和异常是处理程序开发中不可回避的议题,在本地开发中我们往往希望能捕获程序抛出的异常并将其显示打印出来,以便直观的知道程序在哪里出了问题并予以解决,而在线上环境我们不希望将程序错误或异常显示在浏览器中(出于安全考虑),这个时候我们仍然要捕获异常,只不过不是显示到浏览器中,而是记录到日志中,方便日后排查问题. 百牛信息技术bainiu.ltd整理发布于博客园 Laravel当然支持PHP原生的错误和异常处理,但是在此基础上进行了一些封装处理,从而更方便在不同开发环境切换以及对错误和异常的处理.

Java必知必会:异常机制详解

一.Java异常概述 在Java中,所有的事件都能由类描述,Java中的异常就是由java.lang包下的异常类描述的. 1.Throwable(可抛出):异常类的最终父类,它有两个子类,Error与Exception. Throwable中常用方法有: getCause():返回抛出异常的原因.如果 cause 不存在或未知,则返回 null. getMeage():返回异常的消息信息. printStackTrace():对象的堆栈跟踪输出至错误输出流,作为字段 System.err 的值.

python 异常和弹出框

import tkinter.messagebox try: fileContent = open("abnormal.txt") fileContent.close() print("over") #把异常消息赋予一个"ex"变量 except Exception as ex: print(ex) tkinter.messagebox.showinfo("Alert",ex) tkinter.messagebox.askye

python基础--接口与归一化设计、封装、异常、网络编程

1 接口与归一化设计 1.1 归一化概念: 归一化的好处: 1.归一化让使用者无需关心对象的类是什么,只需要知道这些对象都具备某些功能就可以了,这极大降低了使用者的使用难度. 2.归一化使得高层的外部使用者可以不加区分的处理所有接口兼容的对象集合 继承的两种用途 一:继承基类的方法,并且做出自己改变或者扩展(代码重用):实践中,继承的这种用途意义并不很大,甚至常常是有害的.因为它使得子类与基类出现强耦合. 二:声明某个子类兼容于某基类,定义一个接口类(模仿java的Interface),接口类中

异常上报工具:腾讯Bugly

1.腾讯出了一个和umeng差不多的异常上报工具Bugly.(传送门:https://bugly.qq.com/docs/) (1)两者比较明显的区别是,Bugly能比较实时上报异常信息,经过测试基本上几秒就能在后台看到上报的信息,umeng的要等一段时间:(这个是我选择Bugly的主要原因) (2)Bugly感觉比较轻盈,主要方向就是异常上报:umeng感觉比较臃肿,主要方向在后台数据统计上:(个人感觉) 2.Cocos的接入文档就在里面,提供了c++/lua/js三种方式的接入,很简单方便就

最近遇到的异常与错误总结

异常 NumberFormatException  数字格式化异常 ArithmeticException 算术异常 ArrayIndexOutOfBoundsException  数组超出绑定异常:没有输入参数,或输入的参数不够 NullPointerException 空指针异常:使用了未实例化的对象 NoSuchMethodError:main  找不到主方法 ClassCastExeption:A 类转换异常 IllegalThreadStateException:非法的线程状态异常 I