57、只针对异常的情况才使用异常

try {
int i = 0;
while(true)
range[i++].climb();
}catch(ArrayIndexOutOfBoundsException e) { }

在这段程序中,当循环企图访问数组边界之外的元素时,程序抛出异常并结束无限循环。。使用异常以达到终止无限循环,这种模式不仅模糊了代码的意图,而且降低了性能(因为异常模式比标准模式慢的多)。

异常应该只用于异常的情况,不要将它们用于控制流,也不要编写迫使客户端使用控制流的API。

58、对可恢复的情况使用受检异常,对编程错误使用运行时异常

Java语言提供了三种可抛出结构(throwable):「受检的异常」、「运行时异常」和「error」。

  • 「受检的异常」是程序可以处理的异常,如果抛出异常的方法本身不能处理它,那么方法的调用者就应该去处理它,从而使程序恢复运行。例如,喷墨打印机在打印文件时,如果纸用完或者墨水用完,就会暂停打印,等待用户添加打印纸或更换墨盒,如果用户添加了打印纸或更换了墨盒,就能继续打印。

  • 「运行时异常」是程序无法恢复运行的异常,导致这种异常的原因通常是由于执行了错误操作。一旦出现了错误操作,建议终止程序并仔细的debug,因此Java编译器不检查这种异常。

  • 「error」通常是系统出现了不可控的错误,这个错误通常与程序无关,一般不需要处理。 运行时异常和error都是不可恢复的情形。

对于可恢复的情况,使用受检的异常;对于程序错误使用运行时异常。对于自定义的未受检的抛出结构不要继承自Error,它应该是RuntimeException的子类。

59、避免不必要的使用受检的异常

过分使用受检的异常会使API使用起来非常不便,影响API的灵活性。

60、优先使用标准异常

使用标准异常的好处有:使API更易学习和使用、可读性更强和高度的代码重用。

常用的异常:

异常 使用场合
IllegalArgumentException 非null的参数值不正确
IllegalStateException 对于方法调用而言,对象状态不合适
NullPointerException 参数为null
IndexOutOfBoundsException 下标参数越界
ConcurrentModificationException 在禁止并发修改的情况下,检测到对象的并发修改
UnsupportedOperationException 对象不支持用户请求的方法

61、抛出与抽象相对应的异常

当方法传递由低层方法抛出的异常时,方法所抛出的异常与它执行的任务没有明显的关系,这往往使人困惑,也让实现细节污染了高层的API。为了避免这个问题,更高层的实现应该捕获低层的异常,同时抛出按高层抽象进行解释的异常。这种做法被称为「异常转译」。

例如:AbstractSequentialList类中

/**
* Returns the element at the specified position in this list.
*
* @throw IndexOutOfBoundsException if the index is out of range
* ({@code index < 0 || index >= size()}).
*/
public E get(int index) {
ListIterator<E> i = listIterator(index);
try {
return i.next();
} catch (NoSuchElementException e) {
throw new IndexOutOfBoundsException("Index: " + index);
}
}

如果低层的异常对于调试导致高层异常的问题有帮助,可以使用「异常链」将低层的异常传到高层的异常中,并使用高层异常的方法(Throwable.getCause)来获取低层异常。如:

try {
...
} catch (LowerLevelException e) {
throw new HigherLevelException(e);
}

总之,如果不能阻止或处理来自低层的异常,一般使用「异常转译」来保证所抛出的异常适合高层。「异常链」:它允许抛出适当的高层异常,同时又能捕获底层的原因进行失败分析。

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

要为编写的每个方法所能抛出的每个异常建立文档。为每个受检异常提供单独的throws子句,并利用@throws标记记录下抛出每个异常的条件。不要使用throws关键字将未受检的异常包含在方法的声明中。

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

异常的字符串表示法主要是让程序员或域服务人员来分析失败的原因,所以其应该包含尽可能多的失败信息,以便于分析。

为了确保在异常的细节消息中包含足够的能捕获失败的信息,一种做法是在异常的构造器而不是字符串细节消息中引入这些信息。如:

/**
* Constructs an <code>IndexOutOfBoundsException</code> with the
* specified detail message.
* @param lowerBound the lowest legal index value
* @param upperBound the highest index value plus one
* @param index the actual index value
*/
public MyIndexOutOfBoundsException(int lowerBound, int upperBound, int index) {
super("Lower bound: " + lowerBound + ", Upper bound: " + upperBound +
", Index: " + index);
} //使用
....
throw new MyIndexOutOfBoundsException(0, 10, i);

64、努力使失败保持原子性

一般而言,为了能从异常中恢复,失败的方法调用应该使对象保持在被调用之前的状态,称这中方法具有「失败原子性」。

对于不可变对象,它具有失败原子性是显然的。因为对象的状态始终保持一致。

对于可变对象获得失败原子性最常见的方法:在执行操作之前检查参数的有效性,这可以使对象的状态在被修改之前,先抛出适当的异常。如:

public Object pop() {
if(size == 0)
throw new EmptyStackException();
Object result = elements[--size];
....
}

总之,产生任何异常都应该让对象保持在方法调用之前的状态。若违反了这条规则,API文档中应该清楚的指明对象处于什么样的状态。

65、不要忽略异常

用空的catch块来忽略异常,可能会产生灾难性的后果。永远也不要忽略异常。

Effective java笔记(八),异常的更多相关文章

  1. Effective Java笔记一 创建和销毁对象

    Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...

  2. Effective java笔记(二),所有对象的通用方法

    Object类的所有非final方法(equals.hashCode.toString.clone.finalize)都要遵守通用约定(general contract),否则其它依赖于这些约定的类( ...

  3. effective java笔记之单例模式与序列化

    单例模式:"一个类有且仅有一个实例,并且自行实例化向整个系统提供." 单例模式实现方式有多种,例如懒汉模式(等用到时候再实例化),饿汉模式(类加载时就实例化)等,这里用饿汉模式方法 ...

  4. effective java笔记之java服务提供者框架

    博主是一名苦逼的大四实习生,现在java从业人员越来越多,面对的竞争越来越大,还没走出校园,就TM可能面临失业,而且对那些增删改查的业务毫无兴趣,于是决定提升自己,在实习期间的时间还是很充裕的,期间自 ...

  5. Effective java笔记6--异常

    充分发挥异常的优点,可以提高一个程序的可读性.可靠性和可维护性.如果使用不当的话,它们也会带来负面影响. 一.只针对不正常的条件才使用异常 先看一段代码: //Horrible abuse of ex ...

  6. Effective java笔记5--通用程序设计

    一.将局部变量的作用域最小化      本条目与前面(使类和成员的可访问能力最小化)本质上是类似的.将局部变量的作用域最小化,可以增加代码的可读性和可维护性,并降低出错的可能性. 使一个局部变量的作用 ...

  7. Java笔记:异常

    Exception 类的层次 所有的异常类是从 java.lang.Exception 类继承的子类. Exception 类是 Throwable 类的子类.除了Exception类外,Throwa ...

  8. Effective java笔记(四),泛型

    泛型为集合提供了编译时类型检查. 23.不要在代码中使用原生态类型 声明中具有一个或多个类型参数的类或接口统称为泛型.List<E>是一个参数化类,表示元素类型为E的列表.为了提供兼容性, ...

  9. Effective java笔记(九),并发

    66.同步访问共享的可变数据 JVM对不大于32位的基本类型的操作都是原子操作,所以读取一个非long或double类型的变量,可以保证返回的值是某个线程保存在该变量中的,但它并不能保证一个线程写入的 ...

随机推荐

  1. c#多线程介绍(上)

    转载原文:这里是链接内容 转载原文:这里写链接内容 转载原文:这里写链接内容 (重要事情说三遍) 引言 本文主要从线程的基础用法,CLR线程池当中工作者线程与I/O线程的开发,并行操作PLINQ等多个 ...

  2. NEsper Nuget包

    Esper是专门进行复杂事件处理(CEP)的流处理平台,Java版本为Esper,.Net版本为NEsper.Esper & NEsper可以方便开发者快速开发部署处理大容量消息和事件的应用系 ...

  3. Reactive Extensions(Rx) 学习

    Bruce Eckel(著有多部编程书籍)和Jonas Boner(Akka的缔造者和Typesafe的CTO)发表了“反应性宣言”,在其中尝试着定义什么是反应性应用. 这样的应用应该能够: 对事件做 ...

  4. 利用history实现无刷新跳转界面

    看标题是不是感觉很拽的样子,其实没什么啦,也就是时下常说的单页面应用.这种web形式在如今的移动端十分流行,毕竟在移动端频繁得去刷新界面不是很友好,而且还费流量.今天我们要做一个小的app(移动端), ...

  5. Python黑帽编程 2.0 第二章概述

    Python黑帽编程 2.0 第二章概述 于 20世纪80年代末,Guido van Rossum发明了Python,初衷据说是为了打发圣诞节的无趣,1991年首次发布,是ABC语言的继承,同时也是一 ...

  6. Hadoop学习笔记—18.Sqoop框架学习

    一.Sqoop基础:连接关系型数据库与Hadoop的桥梁 1.1 Sqoop的基本概念 Hadoop正成为企业用于大数据分析的最热门选择,但想将你的数据移植过去并不容易.Apache Sqoop正在加 ...

  7. 【大型网站技术实践】初级篇:搭建MySQL主从复制经典架构

    一.业务发展驱动数据发展 随着网站业务的不断发展,用户量的不断增加,数据量成倍地增长,数据库的访问量也呈线性地增长.特别是在用户访问高峰期间,并发访问量突然增大,数据库的负载压力也会增大,如果架构方案 ...

  8. 学习RBAC 用户·角色·权限·表

  9. 基于stm32f4的ucGUI通过外部flash存储汉字库显示任意英文字符和汉字组合(控件可用)

    在做一个用到ucGUI的项目的时候要用到不定的汉字和英文字符,但是ucGUI本身又不支持读取芯片外部flash的字库来显示,于是查了下资料,如下: http://www.cnblogs.com/hik ...

  10. Execute SQL Task 参数和变量的映射

    Execute SQL Task能够执行带参数的SQL查询语句或存储过程(SP),通过SSIS的变量(Variable)对参数赋值.对于不同的Connection Manager,在Task中需要使用 ...