JavaSE06_异常

1 什么是异常

  • 实际工作中,可能会遇到一些突发状况。例如:某个模块用户输入不符合程序要求;程序需要打开某个文件,这个文件可能不存在或者格式不对;读取的数据库可能数据是空的;程序运行时内存或硬盘满了等等
  • 程序运行时遇到的这些突发状况称为异常,英文为exception,意为例外情况
  • 异常指程序运行中出现的不期而至的各种状况,如:文件找不到、网络连接失败、非法参数等
  • 异常发生在程序运行期间,它影响了正常的程序执行流程

2 异常分类

  • 检查性异常:是由用户错误或问题引起的异常,是程序员无法预见的。例如要打开一个不存在的文件时,一个异常就发生了,检查性异常在编译时不能被简单地忽略检查性异常必须处理,如果不处理就会编译失败。
  • 运行时异常(RuntimeException及其子类):是可能被程序员避免的异常,运行时异常可以在编译时被忽略
  • 错误(Error):错误不是异常,是脱离程序员控制的问题。错误在代码中通常被忽略,例如在栈溢出时,一个错误就发生了,在编译时也检查不到

3 异常体系结构

  • Java把异常当做对象来处理,并定义一个基类java.lang.Throwable作为所有异常的超类
  • 在Java API中定义了许多异常类,分为两大类,错误Error和异常Exception在这里插入图片描述

4 Error

  • Error类对象由Java虚拟机生成并抛出,大多数错误与代码编写者所执行的操作无关
  • Java虚拟机运行错误(VirtualMachineError),当JVM不再有继续执行操作所需的内存时,将出现OutOfMemoryError。错误发生时,JVM一般会选择线程终止
  • 虚拟机试图执行应用时,可能发生类定义错误(NoClassDefFoundError)、链接错误(LinkageError)。这些错误是不可查的,因为它们在应用程序的控制和处理能力之外,且绝大多数是程序运行时不允许出现的状况

5 Exception

  • Exception中的重要子类RuntimeException(运行时异常),这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理
  • 运行时异常一般由程序逻辑错误引起,程序应该从逻辑角度尽可能避免这类异常的发生
  • Error通常是灾难性的致命错误,是程序无法控制和处理的,Error出现时,JVM一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能地处理这些异常

6 异常处理机制

  • Java处理异常的五个关键字:trycatchfinallythrowthrows

  • try代码块中是需要进行异常捕获的代码,catch()中是想要捕获的异常类型,finally代码块用于处理善后工作,且无论是否出现异常,总是会执行finally中的语句

    public class TestException {
        public static void main(String[] args) {
            int a = 1;
            int b = 0;
            
            try {		// try代码块为监控区域
                System.out.println(a / b);
            } catch (ArithmeticException e) {	// catch用于捕获异常
                System.out.println("程序出现异常,变量b不能为0");
            } finally {		// 无论是否出现异常,都会执行finally中的代码
                System.out.println("finally");
            }
        }
    }
    
  • 程序中可以没有finallyfinally常用于IO资源关闭

  • catch(Throwable e)可以捕获所有的ErrorException

  • 可以使用多个catch语句层层捕获异常,从上往下的异常类型的范围必须越来越大,不能出现上方的异常类型包含下方类型的情况

    try {
        System.out.println(a / b);
    } catch (Error e) {
        System.out.println("This is an Error");
    } catch (Exception e) {
        System.out.println("This is an Exception");
    } catch (Throwable t) {
        System.out.println("This is a Throwable class");
    } finally {
        System.out.println("finally");
    }
    
  • catch中使用e.printStackTrace()打印出现异常的栈信息

  • throws用于声明一个方法中可能产生的所有异常,用在方法声明后,后跟异常类名。throws不对异常做处理,而是将异常向上传,由方法的调用者处理

    class Math{
        public int div(int i, int j) throws Exception {	// 向上抛出
            int t = i / j;
            return t;
        }
    }
    
    public class TestException {
        public static void main(String[] args) {
            // 调用由throws声明的方法,必须显式捕获该异常
            // 否则必须在main方法中再次声明抛出
            // public static void main(String[] args) throws Exception
            try {
                new Math().div(1, 0);
            } catch (Exception e) {
                System.out.println("出现异常");
            }     
        }
    }
    
  • throw用于抛出一个具体的异常类型,一般在方法中使用,throw出现的地方明确表示会抛出相应的异常

    public class TestException {
        public static void main(String[] args) {
            test(1, 0);
        }
        
        public static test(int a, int b) {
            if (b == 0) {
                // 自行抛出ArithmeticException异常
                // 既可以显式捕获该异常,也可以交给方法调用者处理
                throw new ArithmeticException();
            }
        }
    }
    
  • throwthrows的区别

    • throws出现在方法头,throw出现在方法体
    • throws表示出现异常的一种可能性,但并不一定会发生这些异常;执行throw语句时则一定抛出了某种异常
    • 二者都是消极处理异常的方式,只是抛出或可能抛出异常,真正的异常处理由调用该方法的上层处理

7 自定义异常

  • 使用Java内置的异常类可以描述编程时出现的绝大多数异常情况。除此之外,用户还可以自定义异常,用户的自定义异常类只需继承Exceptin类即可

    public class MyException extends Exception {
        private int detail;
        
        public MyException(int detail) {
            this.detail = detail;
        }
        
        // 打印异常信息
        @Override
        public String toString() {
            return "MyException{" + detail + "}";
        }
    }
    
  • 使用自定义异常类大体分为以下几个步骤:

    • 创建自定义异常类
    • 在方法中通过throw关键字抛出异常对象
    • 如果在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throws关键字向上抛出异常
    • 在方法调用者出捕获并处理异常
    public class TestException {
        // 可能会出现异常的方法
        static void test(int a) throws MyException {
            if (a > 10) {
                throw new MyException(a);	// 抛出自定义异常
            }
        }
        
        public static void main(String[] args) {
            try {	// 捕获并处理自定义异常
                test(11);
            } catch (MyException e) {
                System.out.println("MyException => " + e);	
                // 输出:MyException => MyException{11}
            }
        }
    }
    

8 异常处理经验总结

  • 处理运行时异常时,采用逻辑合理规避异常,同时使用try-catch语句辅助处理
  • 在多重catch块后,可以添加一个catch(Exception)来处理可能会被遗漏的异常
  • 对于不确定的代码,也可以加上try-catch语句处理潜在的异常
  • 尽量去处理异常,而不是简单地调用printStackTrace()去打印输出
  • 具体如何处理异常,要根据不同的业务需求和异常类型来决定
  • 尽量添加finally语句释放占用的资源
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值