异常处理


异常处理的定义

异常定义: 导致程序的正常流程被中断的事件,叫做异常

实例:

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {
    public static void main(String[] args) {
//        文件不存在异常
        File f=new File("LOL.exe");

//        尝试打开LOL.exe,会抛出java.io.FileNotFoundException,如果不处理该异常,会有编译错误
        try {
            new FileInputStream(f);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
    }
}

由于没有找到制定的文件,所以报了一个java.io.FileNotFoundException错误

常见异常

ParseException 解析异常,日期字符串转换为日期对象的时候,有可能抛出的异常

OutOfIndexException 数组下标越界异常
OutOfMemoryError 内存不足
ClassCastException 类型转换异常
ArithmeticException 除数为零
NullPointerException 空指针异常

处理

try catch

1.将可能抛出FileNotFoundException 文件不存在异常的代码放在try里
2.如果文件存在,就会顺序往下执行,并且不执行catch块中的代码
3. 如果文件不存在,try 里的代码会立即终止,程序流程会运行到对应的catch块中
4.e.printStackTrace();会打印出方法的调用痕迹,如此例,会打印出异常开始于文件中的第19行,这样就便于分析到底是哪里出现了异常。

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {
    public static void main(String[] args) {
//        文件不存在异常
        File f=new File("LOL.exe");

//        尝试打开LOL.exe,会抛出java.io.FileNotFoundException,如果不处理该异常,会有编译错误
        try {
            System.out.println("试图打开LOL.exe");
            new FileInputStream(f);
            System.out.println("打开成功");
        } catch (FileNotFoundException e) {
            e.printStackTrace();
            System.out.println("打开失败");
        }
    }
}

使用异常的父类进行catch

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {
    public static void main(String[] args) {
//        文件不存在异常
        File f=new File("LOL.exe");

//        尝试打开LOL.exe,会抛出java.io.FileNotFoundException,如果不处理该异常,会有编译错误
        try {
            System.out.println("试图打开LOL.exe");
            new FileInputStream(f);
            System.out.println("打开成功");
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("打开失败");
        }
    }
}

FileNotFoundException是Exception的子类,所以我们可以用Exception接收FileNotFoundException的异常。

Exception是所有异常类的父类,所以我们可以用Exception接
使用异常的父类进行catch。

多异常捕捉方法

多catch,一个catch接收一个异常

有的时候一段代码会抛出多个异常,比如:

new FileInputStream(f);
Date d = sdf.parse("2016-06-03");

这段代码会抛出一个文件不存在异常FileNotFoundException和解析异常ParseException

解决方法是对两个异常进行分别catch

catch (FileNotFoundException e) {
    System.out.println("d:/LOL.exe不存在");
    e.printStackTrace();
} catch (ParseException e) {
    System.out.println("日期格式解析错误");
    e.printStackTrace();
}
package exception;
import java.io.File;
import java.io.FileInputStream;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {
    public static void main(String[] args) {
//        文件不存在异常
        File f=new File("LOL.exe");

//        尝试打开LOL.exe,会抛出java.io.FileNotFoundException,如果不处理该异常,会有编译错误
        try {
            System.out.println("试图打开LOL.exe");
            new FileInputStream(f);
            System.out.println("打开成功");
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            Date d = sdf.parse("2016-06-03");
        } catch (ParseException e){
            System.out.println("日期解析失败");
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("打开失败");
        }
    }
}

单catch接收多个异常

另外一个办法是把多个异常,放在一个catch中同一捕捉。

catch(FileNotFoundException | ParseException e){

这种方式从JDK7开始支持,好处是使代码更加紧凑,不足之处是:一旦发生异常,不能确定是哪种异常,需要通过instanceof进行具体的异常类型。

package exception;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {
    public static void main(String[] args) {
//        文件不存在异常
        File f=new File("LOL.exe");

//        尝试打开LOL.exe,会抛出java.io.FileNotFoundException,如果不处理该异常,会有编译错误
        try {
            System.out.println("试图打开LOL.exe");
            new FileInputStream(f);
            System.out.println("打开成功");
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            Date d = sdf.parse("2016-06-03");
        }catch(FileNotFoundException | ParseException e){
            if(e instanceof FileNotFoundException)
                System.out.println("文件不存在");
            if(e instanceof ParseException)
                System.out.println("文件解析错误");
        }
    }
}

finally

无论是否出现异常,finally中的代码都会被执行。

package exception;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {
    public static void main(String[] args) {
//        文件不存在异常
        File f=new File("LOL.exe");

//        尝试打开LOL.exe,会抛出java.io.FileNotFoundException,如果不处理该异常,会有编译错误
        try {
            System.out.println("试图打开LOL.exe");
            new FileInputStream(f);
            System.out.println("打开成功");
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
            Date d = sdf.parse("2016-06-03");
        }catch(FileNotFoundException | ParseException e){
            if(e instanceof FileNotFoundException)
                System.out.println("文件不存在");
            if(e instanceof ParseException)
                System.out.println("文件解析错误");
        }finally {
            System.out.println("无论是否出现异常,此代码都将执行");
        }
    }
}

throws

当遇到异常时,通常有两种处理方式:一种是使用catch进行接收;另一种是使用throws抛出,交给其他方法进行catch处理

如下情况是:
主方法调用method1
method1调用method2
method2中打开文件

method2中需要进行异常处理,但是method2不打算处理,而是将这个异常thows抛出,那么method1就会接收到这个异常。处理方法有两种,一种是使用catch处理,另一种是throws抛出。
method1选择本地try catch,将这个异常消化掉,主方法在调用method1时就不需要进行异常处理了。

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        method1();
    }

    static void method1(){
        try {
            method2();
        } catch (FileNotFoundException e) {
            System.out.println("打开失败");
            e.printStackTrace();
        }
    }
    static void method2() throws FileNotFoundException {
        File f=new File("LOL.exe");

        System.out.println("尝试打开LOL.exe");
        new FileInputStream(f);
        System.out.println("成功打开");
    }
}

throw和throws的区别

throws和throw两个关键接近,不过意义不一样,有如下区别:

  1. throws出现在方法声明中,而throw通常出现在方法体中。
  2. throws表示一种出现异常的可能性,并不一定会发生这些异常;throw则是抛出了异常,执行了throw则是抛出了异常,执行了throw则一定抛出了某个异常对象

练习

假设有一个方法 public int method(), 会返回一个整数
在这个方法中有try catch 和 finally.
try 里返回 1
catch 里 返回 2
finally 里 返回3
那么,这个方法到底返回多少?

package exception;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        System.out.println(method());
    }
    
    public static int method(){
        try{
            return 1;
        }catch (Exception e){
            return 2;
        }finally {
            return 3;
        }
    }
}

异常分类

异常分类: 可查异常,运行时异常和错误3种其中,运行时异常和错误又叫非可查异常

可查异常

可查异常: CheckedException
可查异常即必须进行处理的异常,要么try catch住,要么往外抛,谁调用,谁处理,比如 FileNotFoundException
如果不处理,编译器,就不让你通过

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        File f=new File("LOL.exe");

        try {
            new FileInputStream(f);
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
    }
}

运行时异常

运行时异常RuntimeException指: 不是必须进行try catch的异常
常见运行时异常:
除数不能为0异常:ArithmeticException
下标越界异常:ArrayIndexOutOfBoundsException
空指针异常:NullPointerException
在编写代码的时候,依然可以使用try catch throws进行处理,与可查异常不同之处在于,即便不进行try catch,也不会有编译错误
Java之所以会设计运行时异常的原因之一,是因为下标越界,空指针这些运行时异常太过于普遍,如果都需要进行捕捉,代码的可读性就会变得很糟糕。

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        int arr[]=new int[5];

        arr[10]=1;
    }
}

错误

错误Error,指的是系统级别的异常,通常是内存用光了
在默认设置下,一般java程序启动的时候,最大可以使用16m的内存
如例不停的给StringBuffer追加字符,很快就把内存使用光了。抛出OutOfMemoryError
与运行时异常一样,错误也是不要求强制捕捉

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        StringBuffer sb=new StringBuffer();

        for (int i=0;i<Integer.MAX_VALUE;i++){
            sb.append("a");
        }
    }
}

三种分类

总体上异常分为三种:

  1. 错误
  2. 运行时异常
  3. 可查异常

面试题:运行时异常与非运行时异常的区别

运行时异常是不可查异常,不需要进行显式的捕捉
非运行时异常是可查异常,必须进行显式的捕捉,或者抛出,不然会出现编译错误。

Throwable类

Throwable是类,Exception和Error都继承了该类
所以在捕捉的时候,也可以使用Throwable进行捕捉
如图: 异常分ErrorException
Exception里又分运行时异常和可查异常。

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        File f=new File("LOL.exe");

        try {
            new FileInputStream(f);
        } catch (Throwable t) {
            //使用throwable类进行接收
            t.printStackTrace();
        }
    }
}

Throwable练习

在方法声明上,可以抛出指定的异常,比如FileNotFoundException
那么能否抛出Throwable这个类?

这个方法的调用者又该如何处理?

可以抛出Throwable,并且在调用的时候必须进行catch处理。
但是这样的设计不好,因为不知道抛出的类型到底是哪种,无法进行针对性的处理。

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */
public class ex {

    public static void main(String[] args) {
        try {
            method();
        } catch (Throwable throwable) {
            throwable.printStackTrace();
        }
    }

    static void method() throws Throwable{

        File f=new File("LOL.exe");
        new FileInputStream(f);
    }
}

自定义异常

创建自定义异常

一个英雄攻击另一个英雄的时候,如果发现另一个英雄已经挂了,就会抛出EnemyHeroIsDeadException
创建一个类EnemyHeroIsDeadException,并继承Exception
提供两个构造方法

  1. 无参的构造方法
  2. 带参的构造方法,并调用父类的对应的构造方法
class EnemyHeroIsDeadException extends Exception{
    public EnemyHeroIsDeadException(){}

    public EnemyHeroIsDeadException(String msg){
        super(msg);
    }
}

抛出自定义异常

package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.util.Map;

/**
 * @author Aike
 * @create 2021-04-19 15:37
 */


class Hero{
    public String name;
    protected float hp;
    public void attackHero(Hero h) throws EnemyHeroIsDeadException{
        if(h.hp==0){
            throw new EnemyHeroIsDeadException(h.name+"已经挂了,不需要释放技能");
        }
    }

    class EnemyHeroIsDeadException extends Exception{
        public EnemyHeroIsDeadException(){}

        public EnemyHeroIsDeadException(String msg){
            super(msg);
        }
    }

}
public class ex {

    public static void main(String[] args) {
        Hero garen=new Hero();
        garen.name="盖伦";
        garen.hp=666;

        Hero teemo=new Hero();
        teemo.name="提莫";
        teemo.hp=0;

        try {
            garen.attackHero(teemo);
        } catch (Hero.EnemyHeroIsDeadException e) {
            e.printStackTrace();
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值