Effective java笔记(八),异常
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笔记(八),异常的更多相关文章
- Effective Java笔记一 创建和销毁对象
Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...
- Effective java笔记(二),所有对象的通用方法
Object类的所有非final方法(equals.hashCode.toString.clone.finalize)都要遵守通用约定(general contract),否则其它依赖于这些约定的类( ...
- effective java笔记之单例模式与序列化
单例模式:"一个类有且仅有一个实例,并且自行实例化向整个系统提供." 单例模式实现方式有多种,例如懒汉模式(等用到时候再实例化),饿汉模式(类加载时就实例化)等,这里用饿汉模式方法 ...
- effective java笔记之java服务提供者框架
博主是一名苦逼的大四实习生,现在java从业人员越来越多,面对的竞争越来越大,还没走出校园,就TM可能面临失业,而且对那些增删改查的业务毫无兴趣,于是决定提升自己,在实习期间的时间还是很充裕的,期间自 ...
- Effective java笔记6--异常
充分发挥异常的优点,可以提高一个程序的可读性.可靠性和可维护性.如果使用不当的话,它们也会带来负面影响. 一.只针对不正常的条件才使用异常 先看一段代码: //Horrible abuse of ex ...
- Effective java笔记5--通用程序设计
一.将局部变量的作用域最小化 本条目与前面(使类和成员的可访问能力最小化)本质上是类似的.将局部变量的作用域最小化,可以增加代码的可读性和可维护性,并降低出错的可能性. 使一个局部变量的作用 ...
- Java笔记:异常
Exception 类的层次 所有的异常类是从 java.lang.Exception 类继承的子类. Exception 类是 Throwable 类的子类.除了Exception类外,Throwa ...
- Effective java笔记(四),泛型
泛型为集合提供了编译时类型检查. 23.不要在代码中使用原生态类型 声明中具有一个或多个类型参数的类或接口统称为泛型.List<E>是一个参数化类,表示元素类型为E的列表.为了提供兼容性, ...
- Effective java笔记(九),并发
66.同步访问共享的可变数据 JVM对不大于32位的基本类型的操作都是原子操作,所以读取一个非long或double类型的变量,可以保证返回的值是某个线程保存在该变量中的,但它并不能保证一个线程写入的 ...
随机推荐
- for循环中的占位 pass
- Third Day(上班第四天):Android开发环境配置问题相关
换公司新电脑了,重新安装Android开发环境,并配置,具体流程如下:1.百度JDK,访问Oracle官网:http://www.oracle.com/technetwork/java/javase/ ...
- SQL Server 2008, 2008 R2, 2012 and 2014 完全支持TLS1.2加密传输
SQL Server 2008, 2008 R2, 2012 and 2014 完全支持TLS1.2加密传输 微软高兴地宣布所有主流SQL Server客户端驱动和SQL Server发行版已经支持T ...
- [公告]Senparc.Weixin v4.7.0 升级说明(2016-08-08)
本次升级包含了除QY以外所有的类库,升级内容包括: 1.重构Conatainer结构,删除 ItemCollection 属性,直接使用ContainerBag加入到缓存: 2.重构IContaine ...
- Swift语言快速入门
Swift语言快速入门(首部同步新版官方API文档和语法的Swift图书,确保代码可编译,作者专家在线答疑,图书勘误实时跟进) 极客学院 编著 ISBN 978-7-121-24328-8 201 ...
- EditText的小细节
EditText获取焦点时,右边图片的改变例如: <item android:state_window_focused="false" android:drawable=&q ...
- 谈谈StringBuffer和StringBuilder
(1) 速度 在执行速度方面的比较:StringBuilder > StringBuffer > String ①String 是不可变的对象(String类源码中存放字符的数组被声明为f ...
- Security10:授予访问Object的权限
1,将访问Object的权限授予Database Role 或 User 的语法如下 GRANT <permission> [ ,...n ] ON [ OBJECT :: ][ sche ...
- CSS系列:CSS选择器
选择器(selector)是CSS中很重要的概念,所有HTML语言中的标记样式都是通过不同的CSS选择器来控制的.用户只需要通过选择对不同的HTML标签进行选择,并赋予各种样式声明,即可实现各种效果. ...
- Tomcat报java.lang.OutOfMemoryError: Java heap space错误停止运行如何解决
最近开发的一个商业项目,部署完成后,经常出现Tomcat挂掉的现象,报的异常是:java.lang.OutOfMemoryError: Java heap space,上网google了一下,了解了一 ...