二十九、异常处理

目录

①前言:

②常见的运行时异常

③常见的编译时异常

④异常的处理机制

⑤自定义异常


①前言:

1.什么是异常?

        异常是程序在“编译”或者“执行”的过程中可能出现的问题,注意:语法错误不算在异常体系中。

        比如:   数据索引越界异常,空指针异常,日期格式异常,等。

2.为什么要学习异常?

        异常一旦出现,如果没有提前处理,程序就会退出JVM虚拟机而终止。

        研究异常并且避免异常,然后提前处理异常,体现的是程序的安全,健壮性。

3.异常体系

        Error: 系统级别的问题,JVM退出等,代码无法控制。

        Exception: java.lang包下,称为异常类,它表示程序本身可以处理的问题。

        RuntimeException及其子类: 运行时异常,编译阶段不会报错。(空指针异常,数组索引越界异常)

        除RuntimeException之外的所有异常: 编译时异常,编译器必须处理的,否则程序不能通过编译。(日期格式化异常)

4.编译时异常和运行时异常

        javac.exe 编译时异常,是在编译成class文件时必须要处理的异常,也称之为受检异常。

        java.exe 运行时异常,在编译成class文件不需要处理,在运行字节码文件时可能出现的异常。

简单来说:

        编译时异常,就是在编译时就出现的异常;

        运行时异常,就是在运行时出现的异常。

②常见的运行时异常

1.运行时异常

        直接继承自RuntimeException或者其子类,编译阶段不会报错,运行时可能出现的错误。

2.运行时异常示例:

        ①数据索引越界异常ArrayIndexOutOfBoundsException

        ②空指针异常NullPointerException 直接输出没问题,但调用空指针的变量的功能就会报错

        ③类型转换异常ClassCastException

        ④数学操作异常ArithmeticException

        ⑤数字转换异常NumberFormatException

运行时异常:

                一般是程序员业务没考虑就好或者编译逻辑不严谨引起的程序错误。

3.代码演示:

import java.io.FileInputStream;
import java.io.InputStream;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
​
/**异常  程序在编译或运行中出现错误(语法错误不算在异常体系中)
 * Throwable
 * Error  系统级别的错误
 * Exception  程序级别的异常(RuntimeException  运行时异常 ) 编译时异常(受检异常)
 */
public class Exception_Demo1 {
    public static void main(String[] args) throws ParseException {
        //运行时异常
//        //1.数据索引越界异常ArrayIndexOutOfBoundsException
//        int[] arr = {10,21,34};
//        System.out.println(arr[3]);//越界异常
//        //2.空指针异常NullPointerException  直接输出没问题,但调用空指针的变量的功能就会报错
//        String name = null;
//        System.out.println(name);
//        System.out.println(name.length());
//        //3.类型转换异常ClassCastException
//        Object o = 23;
//        String s = (String) o;
//        //4.数学操作异常ArithmeticException
//        int c = 10/0;
//        //5.数据转换异常NumberFormatException
//        String number = "23 aa  bb";
//        Integer it = Integer.valueOf(number);
//        System.out.println(it+1);
    }
}

③常见的编译时异常

1.编译时异常

        除RuntimeException之外的所有异常,编译阶段就报错,必须处理,否则代码不通过。

2.编译时异常的作用是什么?

        担心程序员技术不行,在编译阶段就爆出一个错误,目的在于提醒不要出错;

3.代码演示:

import java.io.FileInputStream;
import java.io.InputStream;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
​
/**
 * 编译时异常的解决方式:
 * 1、抛出异常:只能抛出一个异常
 * 抛出异常规范操作 直接抛出throws Exception{}
 * 抛出异常并不好,如果异常最终抛出去给虚拟机会引起程序死亡
 *2、 监视捕获异常格式 :捕获异常
 * try{
 * 监视可能出现异常的代码
 * }catch(异常类型1 变量){
 * //处理异常
 * }
 * 3、 前两者结合
 *
 * 运行时异常处理方式:
 * 编译阶段不会报错,可以不抛,默认抛上去
 */
public class Exception_Demo1 {
    public static void main(String[] args) throws ParseException {
        //编译时异常
        //简单日期格式化类
//        String date = "2015-01-12  10:23:21";
//        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
//        //解析字符串成为日期对象
//        Date d = sdf.parse(date);
//        System.out.println(d);
        passTime("2022-12-11 12:24:13");
    }
    public static void passTime(String date){
​
        System.out.println("--------------------------------------");
        try {
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
            Date d = sdf.parse(date);
            System.out.println(d);
        } catch (Exception e) {
//            e.printStackTrace();//打印异常栈信息
            System.out.println("出现了解析时间异常");
        }
        try {
            InputStream is = new FileInputStream("E:/meinv.jpg");
        } catch (Exception ex) {
//            ex.printStackTrace();
            System.out.println("没有这个文件,不要骗我");
        }
    }
}

④异常的处理机制

1.编译时异常

        编译时异常是编译阶段就出错的,所以必须处理,否则代码根本无法通过。

2.编译时异常的处理形式有三种:

        ①出现异常直接抛出去给调用者,调用者也继续抛出去。

        ②出现异常自己捕获处理,不麻烦别人;

        ③前两者结合,出现异常直接抛出去给调用者,调用者捕获处理。

3.异常处理方式之一:throws

        throws:用在方法上,可以将方法内部出现的异常抛出去给本方法的调用者处理。

        这种处理方式并不好,发生异常的方法自己不处理异常,如果异常最终抛出去给虚拟机将引起程序死亡。

抛出异常格式:

规范做法:

 

4.异常处理方式之二:try...catch...

        监视捕获异常,用在方法内部,可以将方法内部出现的异常直接捕获处理。

                这种方式,发生异常的方法自己独立完成异常的处理,程序可以继续往下执行。

格式:

 

5.异常处理方式之三:前两者结合

        方法直接将异常通过throws抛出去给调用者

        调用者收到异常后直接捕获处理。

6.代码演示:

import java.io.FileInputStream;
import java.io.InputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
​
/**
    Exception 异常处理方式
 */
public class Exception_Test1 {
    //异常处理方式之二:try...catch...
    public static void main(String[] args) {
        System.out.println("程序开始~~");
        try {
            passTime("2022-12-11 12:24:13");
            System.out.println("程序操作成功~~");
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("程序操作失败~~");
        }
        System.out.println("程序结束~~");
    }
​
    //异常处理方式之一:throws
    public static void passTime(String date) throws Exception{
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        Date d = sdf.parse(date);
        System.out.println(d);
​
        InputStream is = new FileInputStream("E:/meinv.jpg");
    }
}

7.异常处理的总结

        在开发中按照规范来说第三种方式是最好的:底层的异常抛出去给最外层,最外层集中捕获处理。

        实际应用中,只要代码能够编译通过,并且功能能完成,那么每一种异常处理方式似乎也都是可以的。

8.案例:异常处理使代码更稳健的案例

需求:

        键盘录入一个合理的价格为止(必须是数值,值必须大于0)。

分析:

        定义一个死循环,让用户不断的输入价格;

import java.util.Scanner;
​
/**  案例
 *  键盘录入一个合理的价格位置
 *  定义一个死循环,不断输入价格
 */
public class Exception_Test2 {
    public static void main(String[] args) {
        Scanner sc = new Scanner(System.in);
        while (true) {
            try {
                System.out.println("请您输入一个合理的价格");
                String priceStr = sc.nextLine();//接一行数据
                //转换成double类型的价格  判断价格大于0
                double price = Double.valueOf(priceStr);
                if (price >0) {
                    System.out.println("定价:"+price);
                    break;
                }else {
                    System.out.println("价格必须是正数~~");
                }
            } catch (NumberFormatException e) {
                System.out.println("用户输入的数据无效,请您输入合法的数据");
            }
        }
        System.out.println("----------------------------------");
​
        //自定义异常
        try {
            checkAge(34);
        } catch (Exception_Demo2 e) {
            throw new RuntimeException(e);
        }
    }
    public static void checkAge(int age) throws Exception_Demo2 {
        if (age<0 || age>200){
            //抛出去一个异常对象给调用者
            //throw:在方法内部直接创建一个异常对象,并由此点抛出
            //throws:用在方法上面声明上的,抛出方法内部的异常
            throw new Exception_Demo2(age+"IS Illeagal!");
        }else {
            System.out.println("年龄合法,推荐商品给其购买");
        }
    }
}

⑤自定义异常

1.自定义异常的必要?

        java无法为这个世界上全部问题提供异常类。

        如果企业想通过异常的方式来管理自己的某个业务问题,就需要自定义异常类了。

2.自定义异常的好处?

        可以使用异常的机制管理业务问题,如提醒程序员注意;

        同时一旦出现bug,可以用异常的形式清晰的指出出错的地方。

3.自定义异常的分类

        (1)自定义编译时异常

                ①定义一个异常继承Exception;

                ②重写构造器;

                ③在出现异常的地方用throw new自定义对象抛出。

        作用:

                编译时异常是编译阶段就报错,提醒更加强烈,一定需要处理。

代码演示:

/**自定义异常
 * 1、自定义编译异常
 * 1。继承Exception
 * 2.重写构造器
 * 3.在出现异常的地方,用throw new自定义对象抛出
 * 作用:编译时异常编译时就报错,提醒更加强烈,一定需要处理
 *
 * 2、自定义运行异常
 * 1。继承RuntimeException
 * 2.重写构造器
 * 3.在出现异常的地方,用throw new自定义对象抛出
 * 作用:运行时才出现
 *
 *
 * throw:在方法内部直接创建一个异常对象,并由此点抛出
 * throws:用在方法上面声明上的,抛出方法内部的异常
 */
public class Exception_Demo2 extends Exception {
    public static void main(String[] args) {
        try {
            checkAge(-34);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
​
    public static void checkAge(int age)throws Exception{
        if (age < 0 || age > 200){
            /*抛出一个异常对象给调用者
            throw:在方法内部直接创建一个异常对象,并从此点抛出。
            throws:用在方法申明上的,抛出方法内部的异常。
             */
​
            throw new Exception(age+"is illeagal");
        }else {
            System.out.println("年龄合法:推荐商品给其购买~~~");
        }
    }
}

        (2)自定义运行时异常

                ①定义一个异常类继承RuntimeException;

                ②重写构造器;

                ③在出现异常的地方用throw new 自定义对象抛出。

        作用:

                提醒不强烈,编译阶段不报错,运行时才可能出现。

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

晚风残照

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值