异常处理
异常是程序中的一些错误,但并不是所有的错误都是异常,并且错误有时候是可以避免的。
比如说,你的代码少了一个分号,那么运行出来结果是提示是错误 java.lang.Error;如果你用System.out.println(11/0),那么你是因为你用0做了除数,会抛出 java.lang.ArithmeticException 的异常。
异常发生的原因有很多,通常包含以下几大类:
1.用户输入了非法数据。
2.要打开的文件不存在。
3.网络通信时连接中断,或者JVM内存溢出。
这些异常有的是因为用户错误引起,有的是程序错误引起的,还有其它一些是因为物理错误引起的。
要理解Java异常处理是如何工作的,你需要掌握以下三种类型的异常:
编译型异常:在源码编译阶段,直接抛出异常,这种异常必须处理,否则程序无法执行。
运行时异常: 运行时异常是可能被程序员避免的异常。与编译型异常相反,运行时异常可以在编译时被忽略。
错误: 错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。
Exception 类的层次
所有的异常类是从 java.lang.Exception 类继承的子类。
Exception 类是 Throwable 类的子类。除了Exception类外,Throwable还有一个子类Error 。
Java 程序通常不捕获错误。错误一般发生在严重故障时,它们在Java程序处理的范畴之外。
Error 用来指示运行时环境发生的错误。
例如,JVM 内存溢出。一般地,程序不会从错误中恢复。
异常类有两个主要的子类:IOException 类和 RuntimeException 类。
异常处理
解决异常的出现,让程序继续执行,提高程序的容错能力。
两种解决方案:抓捕异常、抛出异常
抓捕异常
针对于可能出现异常的代码,进行抓捕
try{
//可能发生异常的代码
}catch(Exception e){
//异常出现,代码立刻执行catch中的代码
}finally{
//肯定执行的代码
}
catch 不能独立于 try 存在。
在 try/catch 后面添加 finally 块并非强制性要求的。
try 代码后不能既没 catch 块也没 finally 块。
try, catch, finally 块之间不能添加任何代码。
多重捕获
一个 try 代码块后面跟随多个 catch 代码块的情况就叫多重捕获。
try{
// 程序代码
}catch(异常类型1 异常的变量名1){
// 程序代码
}catch(异常类型2 异常的变量名2){
// 程序代码
}catch(异常类型3 异常的变量名3){
// 程序代码
}
抛出异常
编译型异常,如果不想处理,可以直接抛给调用者,由调用者完成处理
throws关键字:
如果一个方法没有捕获到一个编译型异常,那么该方法必须使用 throws 关键字来声明。throws 关键字放在方法签名的尾部。
也可以使用 throw 关键字抛出一个异常,无论它是新实例化的还是刚捕获到的。
finally关键字
finally 关键字用来创建在 try 代码块后面执行的代码块。
无论是否发生异常,finally 代码块中的代码总会被执行。
在 finally 代码块中,可以运行清理类型等收尾善后性质的语句。
try{
// 程序代码
}catch(异常类型1 异常的变量名1){
// 程序代码
}catch(异常类型2 异常的变量名2){
// 程序代码
}finally{
// 程序代码
}
可以写成:
try{
}finally{
}
结果为:30
finally虽然是必须进行的,但是return比它先进行,会先保存return的值在进行finally。
回收垃圾
关闭IO流
自定义异常
用户自己定义的处理异常的类。
定义一个普通的类,继承Exception类
package oupeng.week6.test;
public class MyException extends Exception{
public MyException(String s){
super(s);
}
}
package oupeng.week6.test;
import java.util.Scanner;
public class TestDemo {
public static void main(String[] args) {
Scanner scanner=new Scanner(System.in);
System.out.println("请输入一个数:");
int n=scanner.nextInt();
if (n>100||n<0){
try {
throw new MyException("输入数字必须大于0小于100");
} catch (MyException e) {
throw new RuntimeException(e);
}
}
System.out.println(n);
scanner.close();
}
}
如果继承的异常不是RuntimeException,需要将异常抛出或者捕获异常。