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. sublime text 如何新建,删除,重命名等问文件的快速操作

    引用自: stackoverflow 可以使用插件, Sidebar Enhancements, 按ctrl+shift+p 输入install package回车 搜索该插件后即可完成

  2. Jmeter之参数化

    Jmeter参数化分为两类,一类是在badboy录制脚本时进行参数化,二是再Jmeter里进行参数化 一:badboy录制脚本时进行参数化的步骤 1.脚本录制成功后->在左下角,点击variab ...

  3. 一个哥们看到数据库日志不断增大 [log_reuse_wait_desc]为replication 之后的做法

    一哥们看到数据库日志不断增大 [log_reuse_wait_desc]为replication 之后的做法 一天那个哥们看到数据库日志暴涨,用sys.databases 视图看一下[log_reus ...

  4. 如何在.NET上处理二维码

    在移动设备,网站以及应用程序间传送数据,而使用二维码真是一种较快捷的方法,也避免了蓝牙配对的混乱状况.ZXing.NET是一个开源,多格式1D/2D条码图像处理库的C#实现,ZXing.NET是个相当 ...

  5. C++的性能C#的产能?! - .Net Native 系列《二》:.NET Native开发流程详解

    之前一文<c++的性能, c#的产能?!鱼和熊掌可以兼得,.NET NATIVE初窥> 获得很多朋友支持和鼓励,也更让我坚定做这项技术的推广者,希望能让更多的朋友了解这项技术,于是先从官方 ...

  6. C语言 · 前缀表达式

    问题描述 编写一个程序,以字符串方式输入一个前缀表达式,然后计算它的值.输入格式为:"运算符 对象1 对象2",其中,运算符为"+"(加法)."-&q ...

  7. 《Entity Framework 6 Recipes》中文翻译系列 (13) -----第三章 查询之使用Entity SQL

    翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 3-4使用实体SQL查询模型 问题 你想通过执行Entity SQL语句来查询你的实 ...

  8. .NET 基础 一步步 一幕幕 [注释、命名规则、访问修饰符、数据类型、常量、变量]

    注释.命名规则.访问修饰符.数据类型.常量.变量 话说一个不会写注释的程序猿的不是一个好吃货,我们本篇就从注释开始说起好了. 在C#中有三种注释: 第一种:单行注释  以//开头,后面的就是注释内容 ...

  9. Mysql 备份

    MySQL数据库备份命令   备份MySQL数据库的命令 mysqldump -hhostname -uusername -ppassword databasename > backupfile ...

  10. ScriptTask读取SharePoint上的Excel

    很多情况下,我们会SharePoint 作为文件共享的Server,有时,业务上需要将SharePoint上的Excel文档 Download 到db中.对于搞DB的人来说,难点是如何下载Excel? ...