龙空技术网

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

图灵课堂程序员小陳 218

前言:

今天兄弟们对“尚硅谷java基础笔记”大概比较讲究,你们都需要了解一些“尚硅谷java基础笔记”的相关知识。那么小编也在网摘上收集了一些有关“尚硅谷java基础笔记””的相关内容,希望看官们能喜欢,我们快快来了解一下吧!

文章目录一、前言

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

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

1.概述

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

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

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

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

Java中异常分为两类:

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];    }}

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.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的使用细节:

finally中声明的是一定会被执行的代码,即使catch中有出现异常,try中有return语句,catch中有return语句等情况像数据库连接、输入输出流、网络编程Socket等资源,JVM是不能自动回收的,我们需要自己手动进行资源释放。此时资源释放需要声明在finally中。为了防止释放之前的代码出现异常,导致资源不被释放, 所以放在finally中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("就算你报异常,我也一定要执行!");        }    }}
我们在开发过程中对于运行时异常不用try-catch处理也无法真正意义上解决,我们得去改代码。所以———运行时异常,不用try-catch但是对于编译时异常需要对它try-catch,否则连编译期都过不去,更别说运行了!编译过去了,如果运行正确就ok。如果运行出错,就是将异常延迟到运行期出现!相当于我们使用try-catch结构将编译时异常变成运行时异常!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();//异常    }}

体会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();//异常    }}

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

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

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

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

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

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

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

五、手动抛出异常(throw)

关于异常对象的产生:

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

代码展示:

public class StudentTest {    public static void main(String[] args) {        try {            Student s = new Student();            //数据非法,需要处理异常对象            //调用此方法进入到catch代码块中            s.regist(-1001);        } catch (Exception e) {        	//控制台输出:您输入的数据非法            System.out.println(e.getMessage());        }    }}class Student{    private int id;	    public void regist(int id) throws Exception {        if (id > 0){            this.id = id;        }else {            //手动抛出异常对象            throw new Exception("您输入的数据非法!");        }    }}
六、自定义异常类

我们上述见到的异常都是Java提供好的(官方的),我们当然也可以自己去自定义异常:

如何自定义异常类?

1.自定义类继承于现有的异常类结构:RuntimeException、Exception2.提供全局常量:serialVersionUID 序列版本号,用于标识类3.提供重载的构造器

代码展示:

//自定义异常类public class MyException extends Exception{    static final long serialVersionUID = -7034897190745766939L;    public MyException(String msg) {        super(msg);    }}//学生类class Student{    private int id;    public void regist(int id){        if (id > 0){            this.id = id;        }else {            throw new MyException("不能输入负数");        }    }}//测试类class Test{    public static void main(String[] args) {        try {            Student s = new Student();            s.regist(-1001);        } catch (Exception e) {            System.out.println(e.getMessage());        }    }}
七、异常处理总结

5个关键字几乎就能涵盖异常处理的所有内容了!

面试题:throw和throws区别?

throw表示抛出一个异常对象,生成异常对象的过程。声明在方法体内throws属于异常处理的一种方式,声明在方法的声明处

标签: #尚硅谷java基础笔记