学习Java异常,吃透这篇足够

举报
dfsafdfas 发表于 2021/02/06 14:48:25 2021/02/06
【摘要】 一、前言这篇技术博客是我复习尚硅谷JavaSE教程做的笔记总结,方便大家的学习同时也方便自己。博客内容非复制粘贴,纯手写。如果对你有帮助,欢迎点赞评论收藏!二、异常概述及异常体系结构1.概述:我们在做程序开发时候,都想着把代码写的完美无瑕(不大可能),但是真实情况是在系统运行代码时,仍然会遇到一些问题,不能靠代码避免,比如:客户输入的数据格式读取文件是否存在网络是否始终保持通畅我们把这类问题...

一、前言

这篇技术博客是我复习尚硅谷JavaSE教程做的笔记总结,方便大家的学习同时也方便自己。博客内容非复制粘贴,纯手写。如果对你有帮助,欢迎点赞评论收藏

二、异常概述及异常体系结构

1.概述:

我们在做程序开发时候,都想着把代码写的完美无瑕(不大可能),但是真实情况是在系统运行代码时,仍然会遇到一些问题,不能靠代码避免,比如:

  • 客户输入的数据格式
  • 读取文件是否存在
  • 网络是否始终保持通畅

我们把这类问题,归结为异常!

异常概念:在Java语言中,将程序执行中发生的不正常情况称为异常(异常指的并不是语法错误,语法错了,编译不通过,不会产生字节码文件,根本不能运行)

Java中异常分为两类

  1. Error:java虚拟机无法解决的严重问题(JVM系统内部错误、资源耗尽),一般没办法编写针对性代码进行处理(处理不了
public class ErrorTest {
    public static void main(String[] args) {
        //1.栈溢出java.lang.StackOverflowError
        main(args);
        
        //2.堆溢出:java.lang.OutOfMemoryError
        Integer[] arr = new Integer[1024 * 1024 * 1024];
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  1. Exception:因编程错误或偶尔的外在因素导致的一般性问题,可以使用针对性的代码进行处理(空指针访问,试图读取不存在的文件,网络连接中断,数组角标越界等)

注意:我们平常将Error和Exception都称作广义上的异常,但是由于我们不处理Error,所以我们平时在开发中提到的异常一般指的是Exception,我们说的异常处理指的就是狭义上的异常:Exception,所以此博客主要讲解的异常处理针对的是Exception

2.异常体系结构

捕获错误最理想的是编译期间,但有的错误只在运行期间发生,比如:除数为0、数组下标越界等

Exception分类:编译时异常(需要处理)、运行时异常(一般不处理)

体系结构图如下
异常的顶级父类是 java.lang.Throwable,其下有两个子类:java.lang.Error 与 java.lang.Exception,平常所说的异常指java.lang.Exception 。
在这里插入图片描述

三、异常处理方式(两种)

Java采用的异常处理机制,是将异常处理的程序代码集中在一起,与正常的程序代码分开,使得程序简洁、优雅,并易于维护。其实异常处理并非真正意义上将异常代码改正,修改代码操作还是需要开发人员自己去做!

异常的处理:抓抛模型
过程一(抛):程序在正常执行过程中,一旦出现异常,就会在异常代码出生成一个对应异常类的对象,并将此对象抛出。一旦抛出对象以后,其后的代码就不再执行。
过程二(抓):可以理解为异常处理的方式:①try-catch-finally ②throws

关于异常对象的产生

  • 系统自动生成的异常对象
  • 手动生成一个异常对象,并抛出(throw)

1.方式一:try-catch-finally

使用格式:

	try{
	...... //可能产生异常的代码
	}
	catch(ExceptionName1 e){
	...... //当产生ExceptionName1型异常时的处置措施
	}
	catch(ExceptionName2 e){
	...... //当产生ExceptionName2型异常时的处置措施
	}
	......
	finally{
	...... //无论是否发生异常,都无条件执行的语句
	}
	
	注意:finally为可选结构,是否使用取决于自己
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

代码展示与说明:

/**
 * 说明:
 * 1.finally是可选的
 * 2.使用try将可能出现异常的代码包装起来,在执行过程中,一旦出现异常,就会生成一个对应异常类的对象
 *   根据此对象的类型,去catch中进行匹配
 * 3.一旦try中的异常对象匹配到某一个catch时,就进入catch中进行异常的处理,一旦处理完成就跳出
 *   当前的try-catch结构(在没写finally情况下),继续执行其后的代码
 * 4.catch中的异常类型如果没有子父类关系,则先声明谁都无所谓
 *   catch中的异常类型如果满足子父类关系,则要求子类一定声明在父类的上面。否则报错
 * 5.常用的异常对象处理的方式:①String getMessage() ②printStackTrace()
 * 6.在try中声明的变量,在出了try结构后,不能再被调用
 */
public class ExceptionTest {
    public static void main(String[] args) {
        String str = "123";
        str = "abc";
        int num = 0;
        try {
            num = Integer.parseInt(str); //可能出现异常的代码
            System.out.println("hello------1");
        }catch (NullPointerException e){
            System.out.println("出现空指针异常了,不要急");

        }catch (NumberFormatException e){
            //System.out.println("出现数值转换异常了,不要急");
            //System.out.println(e.getMessage());
            e.printStackTrace();
        }catch (Exception e){
            System.out.println("出现异常了,不要急");
        }
        System.out.println(num);
        System.out.println("hello------2");
    }
}

上述代码中只使用了try-catch结构,并没有使用finally,接下来我们学习一下finally的使用细节:

  1. finally中声明的是一定会被执行的代码,即使catch中有出现异常,
    try中有return语句,catch中有return语句等情况
  2. 像数据库连接、输入输出流、网络编程Socket等资源,JVM是不能自动回收的,
    我们需要自己手动进行资源释放。此时资源释放需要声明在finally中。
    为了防止释放之前的代码有异常,导致资源不被释放, 所以放在finally中
  3. try-catch-finally可以嵌套使用
//不论是catch中有异常,还是catch中有return,我们的finally代码块一定会执行!
public class FinallyTest {
    public static void main(String[] args) {
        try {
            int a = 10;
            int b = 0;
            System.out.println(a / b);
        }catch (ArithmeticException e){
            //e.printStackTrace();
            int[] arr = new int[10];
            System.out.println(arr[10]); //回报空指针异常,但是没有处理
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            System.out.println("就算你报异常,我也一定要执行!");
        }
    }
}
  • 1
  • 18
我们在开发过程中对于运行时异常不用try-catch处理,比如说空指针异常,出现这类异常使用try-catch
也无法真正意义上解决,我们得去改代码。所以———运行时异常,不用try-catch

但是对于编译时异常需要对它try-catch,否则连编译期都过不去,更别说运行了!
编译过去了,如果运行正确就ok。如果运行出错,就是将异常延迟到运行期出现!
相当于我们使用try-catch结构将编译时异常变成运行时异常!
  • 6

体会1:使用try-catch-finally处理编译时异常,使得程序在编译就不在报错但是运行时仍可能报错。相当于我们使用try-catch-finally将一个编译时可能出现的异常延迟到运行时出现。

体会2:开发中,由于运行时异常比较常见,所以我们通常就不针对运行时异常编写try-catch-finally,针对于编译时异常一定要考虑异常的处理(进行代码修改)!

2.方式二:throws

该方式写在方法的声明处,指明此方法执行时可能会抛出的异常类型,一旦当方法体执行时出现异常,仍会在异常代码处生成一个异常类的对象,此对象满足throws后异常类型时就会被抛出。异常代码后续的代码不再执行!

代码演示:

public class ExceptionTest2{
    public static void main(String[] args){
        //如果抛给main方法,就需要try-catch处理了!
        //不能再向上抛出了
        try {
            method2();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void method2() throws IOException{
        method1();
    }

    public static void method1() throws IOException {
        File file = new File("hello.txt"); //文件找不到,会报异常
        FileInputStream fis = new FileInputStream(file);

        int data = fis.read();//异常
        while (data != -1){
            System.out.println((char)data);
            data = fis.read(); //异常
        }
        fis.close();//异常
    }
}
  • 1127

体会:两种异常处理方式的区别?

  • try-catch-finally真正的将异常处理(并不是修改代码修正异常)
  • throws方式(甩锅)只是将异常抛给方法的调用者,并没有真正将异常处理掉!

注意:子类重写的方法抛出的异常类型不大于(<=)父类被重写方法抛出的异常类型

四、如何选择处理异常方式

我们在开发中处理异常以该选择这两种方式中的哪一种呢?

  • 如果父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能够使用throws(原因看上述注意),意味着子类重写的方法中有异常,就必须使用try-catch-finally方式处理!
  • 执行的方法a中,先后又调用了另外的几个方法,这几个方法是递进关系执行的。建议这几个方法使用throws方式进行处理。而执行的方法a可以考虑使用try-catch-finally方式进行处理

二次理解:我们现在的两种异常处理方式,指的是代码执行之前可能会出现问题,我们提前做一个预案,万一出现问题了我们该怎么办!比如:弹出一个提示框。当然我们同时要明白:异常处理并不是真正意义上讲异常解决!异常处理机制不会帮助我们修正代码,还是需要我们自己处理修改异常代码!

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。