基本概念
Java 语言中,将程序执行中发生的不正常情况称为 “异常”。(开发过程中的语法错误和逻辑错误不是异常)
执行过程中所发生的两类异常事件
Error(错误)
基本介绍
Java 虚拟机无法解决的严重问题。如:JVM 系统内部错误、资源耗尽等严重情况。比如:StackOverflowError[栈溢出]和OOM(out of memory),Error 是严重错误,程序会崩溃
Exception
基本介绍
其它因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。例如空指针访问,试图读取不存在的文件,网络连接中断等等。Exception 分为两大类:运行时异常[程序运行时,发生的异常]和编译时异常[编程时,编译器检查的异常]
五种常见运行时异常
-
NullPointerException(空指针异常)
-
ArithmeticException(算数异常)
-
ArrayStoreException(数组下标越界异常)
-
ClassCastException(类型转换异常)
-
NumberFormatException(数字格式不正确异常)
异常处理
基本介绍
异常处理就是当异常发生时,对异常处理的方式。
异常处理的方式
try-catch-finally :
基本介绍:程序员在代码中捕获的异常,自行处理
try{ //代码/可能有异常 }catch(Exception e){ //捕获到异常 //1.当异常发生时 //2.系统将异常封装成 Exception 对象 e, // 传递给catch //3.得到异常对象后,程序员自己处理 //4.主义:如果没有发生异常,catch代码块不执行 }finally{ //1.不管 try 代码块是否有异常发生,始终要执行finally //2.所以,通常将释放资源的代码放在finally }
使用细节:
-
如果异常发生了,则异常发生后面的代码不会执行(即 try 块中的语句不再执行),直接进入到 catch 块
-
如果异常没有发生,则顺序执行 try 的代码块,不会进入到 catch
-
如果希望不管是否发生异常,都执行某段代码(比如关闭连接,释放资源等)则使用 finally 代码块
-
可以有多个 catch 语句,捕获不同的异常(进行不同的业务处理),要求父类异常在后,子类异常在前,比如(Exception 在后,NullPointerException 在前),如果发生异常,只会匹配一个 catch。代码演示如下:
public class TryCatchDetail { public static void main(String[] args) { try { Person01 person01 = new Person01(); person01 = null; System.out.println(person01.getName()); int n1 = 10; int n2 = 0; int res = n1/n2; }catch (NullPointerException e){ System.out.println("空指针异常=:"+e.getMessage()); }catch (ArithmeticException e){ System.out.println("算数异常=:"+e.getMessage()); } catch (Exception e) {//Exception 是父类异常 System.out.println("异常信息:"+e.getMessage()); } finally { } } } class Person01{ private String name = "崽种"; public String getName() { return name; } }
-
可以进行 try-finally 配合使用,这种用法相当于没有捕获异常,因此程序会直接崩掉/退出。应用场景就是执行一段代码,不管是否发生异常,都必须执行某个业务逻辑。代码演示如下:
public class TryCatchDetail02 { public static void main(String[] args) { try { int n1 = 10; int n2 = 0; System.out.println(n1/n2); } finally { System.out.println("执行finally..."); } //因为没有去捕获异常,所以发生异常,在执行完finally后会直接退出不执行后面代码了 System.out.println("程序继续执行..."); } }
throws :
基本介绍:
-
如果一个方法(中的语句执行时)可能生成某种异常,但是并不能确定如何处理这种异常,则此方法应显示声明抛出异常,表明该方法将不对这些异常进行处理,交给调用者(方法)来处理,最顶级的处理者就是JVM
-
在方法声明中用 throws 语句可以声明抛出异常的列表,throws 后面的异常类型可以是方法中产生的异常类型,也可以是它的父类。代码解释如下:
public void f2() throws FileNotFoundException { //创建了一个文件流对象 //分析: //1.这里的异常是 FileNotFoundException 编译异常 //2.可以使用前面讲过的 try-catch-finally 处理 //3.使用 throws ,抛出异常,让调用f2方法的调用者(方法)处理 //4.throws 后面的异常类型可以是方法中的异常类型,也可以是它的父类 //5.throws 关键字后也可以是 异常列表,即可以抛出多个异常 FileInputStream fis = new FileInputStream("d://aa.txt"); }
使用细节:
-
对于编译异常,程序中必须处理,可使用 try-catch 和 throws
-
对于运行时异常,程序中如果没有处理,默认就是 throws 的方式处理
-
子类重写父类方法时,对抛出异常的规定:子类重写父类的方法,所抛出的异常类型要么和父类抛出的异常一致,要么为父类抛出的异常类型的子类型
class Father{ public void method() throws RuntimeException{//异常类型为RuntimeException } } class Son extends Father{ //子类重写父类方法时,对抛出异常的规定:子类重写父类的方法, //所抛出的异常类型要么和父类抛出的异常一致,要么为父类抛出的异常类型的子类型 public void method() throws ArithmeticException{//异常类型为ArithmeticException } }
-
在 throws 过程中,如果有方法 try-catch,就相当于处理异常,就可以不必 throws
throw 和 throws 的区别:
意义 | 位置 | 后面跟的东西 | |
---|---|---|---|
throws | 异常处理的一种方式 | 方法声明处 | 异常类型 |
throw | 手动生成异常对象的关键字 | 方法体中 | 异常对象 |
throws 处理机制图 :
-
try-catch-finally 和 throws 二选一
-
如果程序员,没有显示处理异常,默认 throws 处
补充要点:
public void f1() throws FileNotFoundException{ //分析 //1.因为f2() 方法抛出的是一个编译异常 //2.即这时,就要f1() 方法必须处理这个编译异常 //3.在f1() 中, 要么 try-catch-finally, 或者继续 throws 这个编译异常 f2();//抛出异常 } public static void f2() throws FileNotFoundException { FileInputStream fis = new FileInputStream("d://aa.txt"); }
public void f3() throws ArithmeticException{} public void f4(){ //分析 //1.在 f4() 中调用 f3() 是可以的 //2.原因是 f3() 抛出的是运行异常 //3.在 Java 中,并不要求程序员显示处理,因为有默认机制 }
自定义异常
基本概念
当程序中出现了某些 “错误” ,但该错误信息并没有在 Throwable 子类中描述处理,这个时候可以自己设计异常类,用于描述该错误信息
自定义异常步骤
-
定义类:自定义异常类名(程序员自己写) 继承 Exception 或 RuntimeException
-
如果继承 Exception, 属于编译异常
-
如果继承 RuntimeException, 属于运行异常(一般来说,继承 RuntimeException)
代码演示
public class CustomException { public static void main(String[] args) { int age = 180; //要求范围在 18~120之间,否则抛出一个自定义异常 if (!(age>= 18 && age<=120)){ //这里我们可以通过构造器,设置异常信息 throw new AgeException("年龄需要在18~120之间..."); } System.out.println("您的年龄符合要求,可以蹲监狱咯..."); } } //自定义一个异常类 //1.一般情况下,我们自定义异常继承 RuntimeException //2.即把自定义异常做成 运行时异常,好处是,我们可以使用默认的处理机制 //3.即比较方便 class AgeException extends RuntimeException{ public AgeException(String message) {//构造器 super(message); } }