异常概念和处理机制
什么是异常?
所谓异常就是指在程序运行的过程中发生的一些不正常事件。(如除0溢出,数组下标越界,所要读取的文件不存在);
异常导致的后果?
Java程序的执行过程中如出现异常事件,可以生成一个异常类对象,该异常对象封装了异常事件的信息,其将被提交给Java运行时系统,这个过程称为抛出异常,不处理的话会导致程序直接中断;
如何防止程序中断?
设计良好的程序应该在程序异常发生时提供处理这些异常的方法,使得程序不会因为异常的发生而阻断或产生不可预见的结果;
异常分类
异常类型 |
包括 |
来源 |
处理 |
受查异常 checkedException |
Exception及其子类(不包括 RuntimeException 及其子类) |
由代码控制能力之外的因素导致额运行时错误 |
必须要处理,否则通不过编译 |
非受查异常 uncheckedException |
Error和 RuntimeException 及其子类 |
RuntimeException 一般代表编译错误 |
可以不用处理 |
Java的异常通过两种机制来处理
捕获:try-catch-finally
抛出:throw,throws
try-catch-finally
try{}语句块中放的是要检测的java代码,可能有会抛出异常,也可能会正常执行;
catch(异常类型){}块是当Java运行时系统接收到try块中所抛出的异常对象时,会寻找能处理这一异常的catch块来进行处理(可以有多个catch块);
finally{}块不管系统有没有抛出异常都会去执行,一般用来释放资源。除了在之前执行了System.exit(0);
1 import java.util.Scanner; 2 import java.util.InputMismatchException; 3 public class ExceptionDemo1{ 4 public static void main(String []args){ 5 System.out.println("请输入一个数字"); 6 Scanner input=new Scanner(System.in); 7 int res=0; 8 try{ 9 //return;//添加return之后,还是会执行finally语句块 10 //System.exit(0);//添加之后不会执行finally语句块 11 int number=input.nextInt(); 12 res=10/number; 13 }catch(InputMismatchException e){ 14 //输入字母时错误 15 //错误信息 16 System.out.println(e.getMessage()); 17 //堆栈信息 18 e.printStackTrace(); 19 }catch(ArithmeticException e){ 20 //输入0时错误 21 //错误信息 22 System.out.println(e.getMessage()); 23 //堆栈信息 24 e.printStackTrace(); 25 }catch(Exception e){ 26 //该异常为父类,若不清楚是何异常,可以使用该类 27 //若都使用时,应该先使用子类再用该类 28 System.out.println(e.getMessage()); 29 e.printStackTrace(); 30 } 31 finally{ 32 //释放资源,比如关闭打开的文件,删除一些临时文件等 33 System.out.println("结果为:"+res); 34 } 35 } 36 }
空指针异常
1 public class ExceptionDemo1{ 2 public static void main(String []args){ 3 //testTryFinally("张三");//输出2 end 4 testTryFinally(null);//空指针异常 5 } 6 7 public static void testTryFinally(String name){ 8 try{ 9 System.out.println(name.length()); 10 }finally{ 11 System.out.println("end"); 12 } 13 } 14 }
throw和throws
throw用于手动抛出异常。作为程序员可以再任意位置手动抛出异常;
throws用于在方法上标识要暴露的异常。抛出的异常交由调用者处理;
两者区别:
① throw用在方法内,后面跟上要抛出的异常类对象;
② throws修饰在方法上,告诉调用者此方法可能会抛出异常,后面跟上要抛出的异常类名;
未处理异常
1 public class ExceptionDemo2{ 2 public static void main(String []args){ 3 Bar bar=new Bar(); 4 bar.enter(15); 5 //未打印end 6 System.out.println("end"); 7 } 8 } 9 10 class Bar{ 11 public void enter(int age){ 12 if(age<18){ 13 //受查异常(必须捕获,否则编译不通过)和非受查异常 14 throw new IllegalArgumentException("年龄不合格"); 15 }else{ 16 System.out.println("欢迎光临"); 17 } 18 } 19 }
在调用方法时处理异常
1 public class ExceptionDemo2{ 2 public static void main(String []args){ 3 Bar bar=new Bar(); 4 try{ 5 bar.enter(15); 6 }catch(IllegalArgumentException e){ 7 System.out.println("错误信息:"+e.getMessage()); 8 } 9 System.out.println("end"); 10 } 11 } 12 13 class Bar{ 14 public void enter(int age){ 15 if(age<18){ 16 //受查异常(必须捕获,否则编译不通过)和非受查异常 17 throw new IllegalArgumentException("年龄不合格"); 18 }else{ 19 System.out.println("欢迎光临"); 20 } 21 } 22 }
方法中未处理异常,但调用者可能不知道抛出什么异常,所有在方法上加上throws,用于方法中未处理异常,交由调用者处理
1 public class ExceptionDemo2{ 2 public static void main(String []args){ 3 Bar bar=new Bar(); 4 try{ 5 bar.enter(15); 6 }catch(IllegalArgumentException e){ 7 System.out.println("错误信息:"+e.getMessage()); 8 } 9 System.out.println("end"); 10 } 11 } 12 13 class Bar{ 14 public void enter(int age)throws IllegalArgumentException{ 15 if(age<18){ 16 //受查异常(必须捕获,否则编译不通过)和非受查异常 17 throw new IllegalArgumentException("年龄不合格"); 18 }else{ 19 System.out.println("欢迎光临"); 20 } 21 } 22 }
受查异常必须要捕获,否则编译不通过
1 public class ExceptionDemo2{ 2 public static void main(String []args){ 3 Bar bar=new Bar(); 4 //try{ 5 bar.enter(15); 6 /*}catch(IllegalArgumentException e){ 7 System.out.println("错误信息:"+e.getMessage()); 8 }*/ 9 System.out.println("end"); 10 } 11 } 12 13 class Bar{ 14 public void enter(int age) { 15 if(age<18){ 16 //受查异常(必须捕获,否则编译不通过)和非受查异常 17 //throw new IllegalArgumentException("年龄不合格");//非受查异常 18 throw new Exception("年龄不合格");//受查异常 19 }else{ 20 System.out.println("欢迎光临"); 21 } 22 } 23 }
当抛出的是受查异常时,且方法中不自己捕获异常,必须加上throws,否则编译不通过
1 public class ExceptionDemo2{ 2 public static void main(String []args){ 3 Bar bar=new Bar(); 4 try{ 5 bar.enter(15); 6 }catch(Exception e){ 7 System.out.println("错误信息:"+e.getMessage()); 8 } 9 System.out.println("end"); 10 } 11 } 12 13 class Bar{ 14 public void enter(int age)throws Exception{ 15 if(age<18){ 16 //受查异常(必须捕获,否则编译不通过)和非受查异常 17 //throw new IllegalArgumentException("年龄不合格");//非受查异常 18 throw new Exception("年龄不合格");//受查异常 19 }else{ 20 System.out.println("欢迎光临"); 21 } 22 } 23 }
自定义异常
常见异常:
非受查异常RuntimeException
受查异常IOException,SQLException,ClassNotFoundException
自定义异常:Java提供的异常体系不可能预见所有希望加以报告的错误;
自定义异常类必须从已有的异常类继承:用的最多的是Exception;
建立新的异常类型最简单的方法就是让编译器产生默认构造方法;
对异常来说,最重要的部分就是它的类名;
可以为异常类定义一个接受字符串参数的构造方法,字符串参数描述异常信息;
1 public class ExceptionDemo3{ 2 public static void main(String []args){ 3 Bar bar=new Bar(); 4 try{ 5 bar.enter(15); 6 }catch(AgeLessThanEighteenException e){ 7 System.out.println("错误信息:"+e.getMessage()); 8 } 9 System.out.println("end"); 10 } 11 } 12 13 //自定义异常 14 class AgeLessThanEighteenException extends Exception{ 15 private String message;//描述异常信息 16 public AgeLessThanEighteenException(String message){ 17 this.message=message; 18 } 19 //重写getMessage()方法 20 public String getMessage(){ 21 return message; 22 } 23 } 24 25 class Bar{ 26 public void enter(int age)throws AgeLessThanEighteenException{ 27 if(age<18){ 28 throw new AgeLessThanEighteenException("年龄不合格"); 29 }else{ 30 System.out.println("欢迎光临"); 31 } 32 } 33 }