一、概念:
如果某个方法不能按照正常的途径完成任务,就可以通过另一种路径退出方法。在这种情况下会抛出一个封装了错误信息的对象。此时,这个方法会立刻退出同时不返回任何值。另外,调用这个方法的其他代码也无法继续执行,异常处理机制会将代码执行交给异常处理器。
二、异常分类:
- Throwable是Java 语言中所有错误或异常的超类。下一层分为Error和Exception
- Error类是指java运行时系统的内部错误和资源耗尽错误。应用程序不会抛出该类对象。如果出现了这样的错误,除了告知用户,剩下的就是尽力使程序安全的终止。Exception()
- Exception又有两个分支,一个是运行时异常RuntimeException,一个是CheckedException。RuntimeException如:NullPointerException、ClassCastException;一个是检查异常CheckedException,如I/O错误导致的IOException、SQLException。RuntimeException是那些可能在Java 虚拟机正常运行期间抛出的异常的超类。如果出现RuntimeException,那么一定是程序员的错误.检查异常CheckedException:一般是外部错误,这种异常都发生在编译阶段,Java编译器会强制程序去捕获此类异常,即会出现要求你把这段可能出现异常的程序进行try catch,该类异常一般包括几个方面:
1.试图在文件尾部读取数据
2.试图打开一个错误格式的URL
3.试图根据给定的字符串查找class对象,而这个字符串表示的类并不存在
三、异常的处理方式
抛出异常有三种形式,一是throw,一个throws,还有一种系统自动抛异常。
throw new NumberFormatException();
int test(int a,int b)throws Exception;
try catch
四、Throw和throws的区别
- throws用在函数上,后面跟的是异常类,可以跟多个;而throw用在函数内,后面跟的是异常对象。
- throws用来声明异常,让调用者只知道该功能可能出现的问题,可以给出预先的处理方式;throw抛出具体的问题对象,执行到throw,功能就已经结束了,跳转到调用者,并将具体的问题对象抛给调用者。也就是说throw语句独立存在时,下面不要定义其他语句,因为执行不到。
- throws表示出现异常的一种可能性,并不一定会发生这些异常;throw则是抛出了异常,执行throw则一定抛出了某种异常对象。
- 两者都是消极处理异常的方式,只是抛出或者可能抛出异常,但是不会由函数去处理异常,真正的处理异常由函数的上层调用处理。
五、代码实现时异常注意事项
1.Java类库中定义的一类RuntimeException可以通过预先检查进行规避,而不应该通过catch来处理,比如:IndexOutOfBoundsException,NullPointerException等等。说明:无法通过预检查的异常除外,如在解析一个外部传来的字符串形式数字时,通过catchNumberFormatException来实现。
正例:if(obj != null) {...}
反例:try { obj.method() } catch(NullPointerException e){...}
2.异常不要用来做流程控制,条件控制,因为异常的处理效率比条件分支低。
3.对大段代码进行try-catch,这是不负责任的表现。catch时请分清稳定代码和非稳定代码,稳定代码指的是无论如何不会出错的代码。对于非稳定代码的catch尽可能进行区分
异常类型,再做对应的异常处理。
4.捕获异常是为了处理它,不要捕获了却什么都不处理而抛弃之,如果不想处理它,请将该异常抛给它的调用者。最外层的业务使用者,必须处理异常,将其转化为用户可以理解的内容。
5.有try块放到了事务代码中,catch异常后,如果需要回滚事务,一定要注意手动回滚事务。
6.不能在finally块中使用return,finally块中的return返回后方法结束执行,不会再执行try块中的return语句。
7.防止NPE,是程序员的基本修养,注意NPE产生的场景:
1)返回类型为基本数据类型,return包装数据类型的对象时,自动拆箱有可能产生NPE。
反例:publicintf(){ returnInteger对象},如果为null,自动解箱抛NPE。
2)数据库的查询结果可能为null。
3)集合里的元素即使isNotEmpty,取出的数据元素也可能为null。
4)远程调用返回对象时,一律要求进行空指针判断,防止NPE。
5)对于Session中获取的数据,建议NPE检查,避免空指针。
6)级联调用obj.getA().getB().getC();一连串调用,易产生NPE。
正例:使用JDK8的Optional类来防止NPE问题。
8.定义时区分unchecked/ checked异常,避免直接抛出new RuntimeException(),更不允许抛出Exception或者Throwable,应使用有业务含义的自定义异常。推荐业界已定义过的自定义异常,如:DAOException/ ServiceException等。
9.在处理必须关闭的资源时,使用try-with-resources语句替代try-finally语句。生成的代码更简洁,更清晰,并且生成的异常更有用。try-with-resources语句在编写必须关闭资源的代码时会更容易,也不会出错,而使用try-finally语句实际上是不可能的。、
反例:
staticvoidcopy(Stringsrc,Stringdst)throwsIOException{
InputStreamin=newFileInputStream(src);
try{
OutputStreamout=newFileOutputStream(dst);
try{
byte[]buf=newbyte[BUFFER_SIZE];
intn;
while((n=in.read(buf))>=0)
out.write(buf,0,n);
}finally{
out.close();
}
}finally{
in.close();
}
}
正例:
staticvoidcopy(Stringsrc,Stringdst)throwsIOException{
try(InputStreamin=newFileInputStream(src);
OutputStreamout=newFileOutputStream(dst)){
byte[]buf=newbyte[BUFFER_SIZE];
intn;
while((n=in.read(buf))>=0)
out.write(buf,0,n);
}
}