异常处理
1、异常概述与异常体系结构
2、常见异常
3、异常处理机制一:try-catch-finally
4、异常处理机制二:throws
5、手动抛出异常对象:throw
6、用户自定义异常类
异常处理类继承关系
java.lang.Object
java.lang.Throwable
java.lang.Exception
通常父异常类型使用Exception,常用的异常类型都是其子类,
可以放在最后一个catch中,用于捕获容易忽略的异常类型
异常:在Java语言中,将程序执行中发生的不正常情况称为异常
注意:开发过程中语法错误和逻辑错误不是异常
Java程序在执行过程中所发生的异常问题可分为两类
1、Error:Java虚拟机无法解决的严重问题
JVM系统内部错误
资源耗尽等严重情况
2、Exception:因编程错误或偶然的外在因素导致的一般性问题
可以使用针对性的代码进行处理
空指针访问
试图读取不存在的文件
网络连接中断
数组角标越界
java.lang.Throwable
java.lang.Error : 错误,不编写针对性的代码进行处理
java.lang.Exception:异常,使用针对性的代码进行处理
编译时异常 非RuntimeException 在执行javac.exe命令时可能出现的异常
运行时异常 RuntimeException 在执行java.exe命令时可能出现的异常
解决办法:
1、遇到错误就终止程序的运行
2、在编写程序时,考虑到错误的检测、错误消息的提示、以及错误的处理
捕获错误最理想的是在编译期间,但有的错误只有运行时才会发生
编译时异常和运行时异常
1、运行时异常
1)编译器不要求强制处置的异常,一般是指编程时的逻辑错误
java.lang.RuntimeException类及它的子类都是运行时异常
2)对于这类异常,可以不做处理,因为这类异常很普遍
若全处理可能会对程序的可读性和运行效率产生影响
2、编译时异常
1)编译器要求必须处理的异常
2)编译器要求Java程序必须捕获或声明所有编译时异常
常见异常
----运行时异常------
java.lang.RuntimeException //运行时异常
ClassCastException//类型转换异常
ArrayIndexOutOfBoundsException//数组下标越界
NullPointerException//空指针异常
ArithmeticException//算术异常
NumberFormatException//数据格式错误
InputMismatchException//输入类型不匹配异常
-----编译时异常--------
java.io.IOExeption
FileNotFoundException//文件找不到
EOFException
java.lang.ClassNotFoundException//找不到类
java.lang.InterruptedException//终端
java.io.FileNotFoundException
java.sql.SQLException
Java异常处理
Java采用的异常处理机制,是将异常处理的程序代码集中在一起,
与正常的程序代码分开,使得程序简洁、优雅、并易于维护
Java提供的异常处理的抓抛模型
Java程序的执行过程中如果出现异常,会生成一个异常类对象
该异常对象将被提交给Java运行时系统,这个过程称为抛出(throw)异常
Java提供的是异常处理的方式:抓抛模型
过程一:“抛”
程序在正常的执行过程中,一旦出现异常,
就会在异常代码处生成对应异常类的对象,并将对象抛出
对象一旦抛出,在没有处理异常的情况下,程序就不再向下执行
关于异常类的对象的生成?分为两种情况
1、系统自动生成异常类的对象
2、我们可以根据程序的需要手动的创建一个异常类的对象
并抛出(throw)
写法:在方法内使用"throw+异常类的对象"
过程二:“抓”
理解为异常处理的方式:
方式一:try-catch-finally
方式二:throws+异常类型
异常对象的生成
1、自动生成
由虚拟机自动生成,程序运行过程中,虚拟机检测到程序发生了问题,
如果在当前代码中没有找到相应的处理程序,就会在后台自动创建一个
对应异常类的实例对象并抛出——自动抛出
2、手动创建
由开发人员手动创建
Excepiton excepiton = new ClassCastException()
创建好的异常对象不抛出对程序没有任何影响,和创建一个普通对象一样
如果一个方法内抛出异常,没有异常对象会抛给调用者方法中处理。
如果异常没有在调用者方法中处理,它继续被抛给这个调用方法的上层方法。
这个过程将一直继续下去,直到异常被处理,这一过程称为捕获异常
如果一个异常回到main()方法,并且main()也不处理,则程序运行终止
程序员通常只能处理Exception,而对Error无能为力
Java异常处理的方式
方式一:try-catch-finally
方式二:throws+异常类型
异常处理机制一:
try-catch-finally
try{
...... //可能产生异常的代码
}
catch( ExceptionName1 e ){
...... //当产生ExceptionName1型异常时的处置措施
}
catch( ExceptionName2 e ){
...... //当产生ExceptionName2型异常时的处置措施
}
[ finally{
...... //无论是否发生异常,都无条件执行的语句
} ]
------
1、finally是可选的
2、try中包含的是可能存在异常的代码,一旦执行出现异常
就会声明相应的异常类的对象,此对象在catch结构中进行匹配,
一旦匹配。就进入catch中进行异常处理。处理完以后,
就不会再匹配后续的catch结构。而是退出try-catch结构,
继续执行其后的代码
3、多个catch中的异常类型如果满足子符类关系,谁上谁下都可以
4、常见的异常处理方式
方法一:getMessage()//获取错误信息
方法二: printStackTrace();//打印错误堆栈信息
5、try结构中声明的变量,在出了try结构以后就不可以被调用
6、try-catch-finally结构可以嵌套
理解:
1、使用异常处理的方式来处理编译时异常,
处理完以后相当于把编译时异常延迟到运行时才可能出现
2、程序中,如果可能报编译时异常,则一定需要进行异常的处理。
如果是运行时异常,则可以不进行处理。
‘
面试题
1、区分final\finally\finalize
2、区分throws、throw
------
try
捕获异常的第一步是用try{……}语句块选定捕获异常的范围
将可能出现异常代码放在try语句块中
catch(Exceptiontype e)
在catch语句块中是对异常对象进行处理的代码,每个try语句块可以伴随
一个或多个catch语句,用于处理可能产生的不同类型的异常对象
----如果明确知道产生的是何种异常,可以用该异常类作为catch的参数
也可以用其父类作为catch的参数
finally
捕获异常的最后一部是通过finally语句为异常处理提供一个统一的出口
使得在控制流转到程序的其它部分以前,能够对程序的状态作统一的管理
无论在try代码块中是否发生了异常事件,catch语句是否执行
catch语句是否有异常,catch语句中是否有return
finally块中的语句都会被执行
finally语句和catch语句是任选的
------
@Test
public void testMethod() {
int result = method();
System.out.println(result);
}
public int method() {
try{
int i = 10;
int j =10;
System.out.println(i / j);
return 1;
}catch(ArithmeticException e) {
e.printStackTrace();
return 2;
}finally {
System.out.println("我是要被执行的代码");
return 3;
}
}
/// 结果为
1
我是要被执行的代码
3
finally的理解
1.一定会被执行的代码声明在finally中
2.不管try或catch中 是否有return,是否有没有处理的异常等情况。
finally中的代码都一定要被执行
------
异常处理机制二:声明抛出异常
声明抛出异常是Java中处理异常的第二种方式
如果一个方法可能生成某种异常,但是并不能确定如何处理这种异常
则此方法已应该显示地声明抛出异常,表明该方法将不对这些异常进行处理,
而由该方法的调用者负责处理
在方法声明中用throws语句可以声明抛出异常的列表,
throws后面的异常类型可以是方法中产生异常类型,也可以是它的父类
-------------
异常处理的方式二:使用throws关键字
说明:
1、在方法的声明处使用“throws+异常类型”的方式抛出异常
2、try-catch-finally的方式,可以理解为从根本上处理异常,程序可以继续执行
throws针对具体的使用此方式来说,采用抛出的方式处理了异常
但是没有从根本上解决此异常,只是抛给了方法的调用者,
方法的调用者也得考虑如何处理此异常
3、如何选择使用try-catch-finally还是throws?
1)如果代码中涉及到资源的关闭(比如:流的关闭、数据库连接的关闭,Socket的关闭),
必须使用try-catch-finally来处理异常
2)父类被重写的方法没有使用throws的方式处理异常
则子类重写的方法也不能使用throws的方式处理异常
3)在一个方法a中先后调用了方法b和方法c。通常方法b和方法c如果有异常的话
会使用throws的方式处理异常。而在方法a中可以考虑使用try-catch-finally的方式处理
-------------
用户如何自定义异常类
1、需要继承于现有的异常类
2、提供一个序列版本号:serialVersionUID。用于唯一的标识当前的类
3、提供重载的构造器