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. MVC学习-http://www.w3school.com.cn/

    连接字符串: <add name="MovieDBContext" connectionString="Data Source=|DataDirectory|\Mo ...

  2. Change the Target Recovery Time of a Database (SQL Server) 间接-checkpoints flushcache flushcache-message

    Change the Target Recovery Time of a Database (SQL Server) 间接checkpoints   flushcache flushcache-mes ...

  3. 微软开源 WCF 分布式服务框架,并入 .NET 基金会项目

    微软北京时间2015.5.20 在其 .NET Foundation GitHub 开源项目页中开放了 WCF 分布式服务框架的代码.WCF突然之间成为一个热门话题,在各大网站上都有不同的报道:dot ...

  4. iOS block种类和切换

    block 分为三种 NSGlobalBlock,NSStackBlock, NSMallocBlock. NSGlobalBlock:类似函数,位于text段: NSStackBlock:位于栈内存 ...

  5. Java static 的一两点使用

    这篇文章是关于Java static关键字的使用,主要会介绍以下的内容: static 的概念 static的各种应用 总结 static 是什么 static 顾名思义是静态的意思.与this相对, ...

  6. windows命令——explorer

    转至http://www.cnblogs.com/ymind/archive/2012/03/30/explorer-command-args.html 今天才知道,explorer原来可以这样用, ...

  7. Qt on Android 核心编程

    Qt on Android 核心编程(最好看的Qt编程书!CSDN博主foruok倾力奉献!) 安晓辉 著   ISBN 978-7-121-24457-5 2015年1月出版 定价:65.00元 4 ...

  8. JavaScript中闭包之浅析解读

    JavaScript中的闭包真心是一个老生常谈的问题了,最近面试也是一直问到,我自己的表述能力又不能完全支撑起来,真是抓狂.在回来的路上,我突然想到了一个很简单的事情,其实我们在做项目时候,其实就经常 ...

  9. C++中static数据成员详解

        本文和大家分享的主要是c++中static数据成员的相关用法及源码示例,希望能帮助大家更好的学习C++. static(静态存储)数据成员 StaticTest.cpp : 定义控制台应用程序 ...

  10. 前端MVVM框架avalon揭秘 - 双向绑定原理

    avalon大家可能不熟悉,但是Knockout估计或多或少听过用过,那么说说KO的几个概念 监控属性(Observables)和依赖跟踪(Dependency tracking) 声明式绑定(Dec ...