异常:程序执行过程中的不正常行为。异常是发生在程序执行过程中,遇到某些代码执行不下去,称之为异常。语法的错误并不属于异常,在编译阶段就会进行报错。
int[] num=new int[3];
System.out.println(num[3]);
int[] num=new int[3];
num=null;
System.out.println(num[1]);
int[] num=new int[3];
num=null;
System.out.println(num[3]);
异常的出现是具有逻辑顺序的,面对空指针异常和越界异常,在程序运行时,只要碰到异常,通常会选择第一个异常进行抛出,终止程序运行。
在Java中,异常也是类,一切皆对象。程序所抛出的异常,抛出的就是异常对象,只不过该对象是由JVM产生并返回给程序的!
异常处理的5个主要关键字:try catch finally throw throws
try{
//可能产生的异常放在try代码块中
}[catch(异常类型 异常对象)][0...N]{
//若捕获到了相应类型的异常对象,如何处理
//放在catch代码块中
}[finally]{
//无论是否有异常产生或者是否处理异常,最终一定会执行的代码放在finally代码块中
//finally中的代码块一定会被执行到
}
try{
}catch(异常类型 异常对象){
}catch....{
}
a. 当没有异常产生时,不走catch块,程序正常执行结束
b. 当异常产生,且被正确的捕获到,则走相应的catch代码块,try代码块中从出现异常之后的代码不再执行;但是异常体系之后的代码会正常执行!
有异常处理之后,保证程序抛出异常之后,正确捕获该异常,就可以使得异常体系结构之后的代码继续执行。
c. 关于catch
块捕获异常的说明
当try中可能会产生多种异常时,可以使用多个代码块进行捕获;
也可以捕获异常的父类,通过向上转型来进行异常捕获(不推荐,具体异常不明确,只要不是error,都可以进行捕获)
输出异常产生的原因及位置,调用异常对象的printStackTrace()
方法,异常出现时,由JVM产生异常对象并返回给程序。
try{
//异常代码
}catch(Exception e){
e.printStackTrace();
}
try{
//异常代码
num=null;
System.out.println(num[100]);
System.out.println("其他代码");
}catch(NullPointerException e){
System.out.println("捕获到空指针异常");
e.printStackTrace();
}catch(Exception e){
e.printStackTrace();
}
System.out.println("异常之后代码");
try{
}catch(异常类型 异常对象){
}finally{
无论是否产生异常,都会执行的代码块
}
总结:
所有关于资源关闭的操作,都会放在finall代码块中,文件的关闭,数据库连接的关闭,统一都放在finally代码块中,确保该资源一定会被关闭。
正常编码,finally一定会正常执行;若出现导致JVM进程直接退出的情况,finally才不会执行,eg:System.exit(0);
若出现异常,不会进行处理,将异常抛回给调用者处理。
若在整个调用过程中没有一个位置进行异常处理,最终这个异常对象就会抛回给JVM,整个程序退出。
关键点:
(1)throws 关键字在方法列表之后进行定义;
(2)throws 可以抛出多个异常,多个异常使用‘,’
进行分隔,若有父子关系,只需要抛出父类异常即可;
(3)throws 抛出的必须是Exception以及其子类。
特殊说明:
若throws抛出的是受查异常,则调用者必须进行显式的异常处理(要么使用try catch捕获,要么继续通过throws向上抛出)
clone抛出CloneNotSupportedException (受查异常) 代码示例
对于异常未作处理,相当于throws向外抛出的情况
使用try catch对自己产生的异常进行捕获处理
总结:
无论是哪种方式产生的对象(JVM产生还是自己throw new的对象)与异常产生的方式无关。
是否需要进行显式处理,需要判断异常对象是受查异常还是非受查异常;只要是受查异常就必须进行显式异常处理!!!
以简单用户登录为例:
public class MyException {
static String userName="张三";
static String passWord="123";
public static void main(String[] args) throws UserNameException {
Scanner sc=new Scanner(System.in);
System.out.println("请输入您的用户名:");
String user=sc.nextLine();
System.out.println("请输入您的密码:");
String pass=sc.nextLine();
login(user,pass);
}
public static void login(String userName,String passWord) throws UserNameException {
if(!userName.equals(MyException.userName))
{
//抛出用户名错误异常
throw new UserNameException("用户名错误!");
}
if(!passWord.equals(MyException.passWord))
{
//抛出密码错误异常
throw new PassWordException("密码错误");
}
System.out.println("登录成功!");
}
}
//用户名属于受查异常
class UserNameException extends Exception{
//只需要提供构造方法即可
public UserNameException(String msg){
super(msg);
}
}
//密码错误属于非受查异常
class PassWordException extends RuntimeException{
public PassWordException(String msg){
super(msg);
}
}