300字范文,内容丰富有趣,生活中的好帮手!
300字范文 > 黑马程序员全套Java教程_Java基础教程_异常(含扩展)(二十三)

黑马程序员全套Java教程_Java基础教程_异常(含扩展)(二十三)

时间:2021-04-05 18:01:49

相关推荐

黑马程序员全套Java教程_Java基础教程_异常(含扩展)(二十三)

黑马程序员全套Java教程_Java基础教程_异常(含扩展)(二十三)

1.1 异常概述与异常体系结构1.2 JVM遇到异常时的默认处理方案1.3 异常处理1.4 异常处理之try……catch……1.5 Throwable的成员方法1.6 编译时异常和运行时异常的区别1.7 异常处理之throws1.8 自定义异常

1.1 异常概述与异常体系结构

在开发过程中,即是我们把代码写得尽善尽美,在系统运行过程中仍然会遇到一些问题,因为很多问题不是靠代码能够避免的,如:客户输入数据的格式读取文件是否存在网络是否始终保持通畅等。异常:在Java语言中,将程序执行中发生的不正常情况称为“异常”。(开发过程中的语法错误和逻辑错误等代码问题不是异常)Java程序执行时发生的异常事件的分类:

(1)Error:Java虚拟机无法解决的严重问题(程序执行分为编译和运行两个过程,运行的时候特别需要使用JVM)。如:JVM系统内部错误、资源耗尽(比如StackOverflowError(栈溢出,如下例:递归调用导致的栈溢出)OOM(堆溢出))等严重情况。一般不(无法)编写针对性的代码进行处理。

(2)Exception(狭义上的异常,我们所说的异常处理通常就是指这个,不包括Error):其他因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。如:

(1)空指针访问

(2)试图读取不存在的文件

(3)网络连接中断

(4)数组角标越界

//资源耗尽的两种情况//1、递归调用导致的栈溢出java.lang.StackOverflowErrorpublic class ErrorTest {public static void main(String[] args) {main(args);}}//2、new的空间过大,堆溢出java.lang.OutOfMemoryErrorpublic class ErrorTest {public static void main(String[] args) {Integer[] arr = new Integer[1024*1024*1024];}}

异常的两种解决方法:一是遇到错误就终止程序的执行(默认)。二是由程序员在编写程序时,就考虑到错误的检测、错误消息的提示,以及错误的处理。异常体系结构:

Error和Exception是两个类,我们可以查看api,它们的父类是Throwable(为顶级父类)。

(1)java.lang.Error:An Error is a subclass of Throwable that indicates serious problems that a reasonable application should not try to catch. Most such errors are abnormal conditions. The ThreadDeath error, though a “normal” condition, is also a subclass of Error because most applications should not try to catch it.

Error是Throwable的一个子类,它表示合理的应用程序不应该捕捉的严重问题。大多数这样的错误都是异常情况。

除了ThreadDeath,其他错误基本都是以*****Error的格式命名(如java.lang.StackOverflowError和java.lang.OutOfMemoryError),ThreadDeath错误虽然是“正常”情况,但也是错误的一个子类,因为大多数应用程序不应该尝试捕捉它。

综上,对于Error,我们一般不编写针对性的代码进行处理。

(2)java.lang.Exception:上图Exception类下,红色为编译时异常,蓝色为运行时异常,关于两者区别可转到本文1.6。

1.2 JVM遇到异常时的默认处理方案

运行下列程序:

public class ExceptionDemo2 {public static void main(String[] args) {System.out.println("开始");method();System.out.println("结束");}public static void method() {int[] arr = {1, 2, 3};System.out.println(3);}}

控制台会输出结果:

开始Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 3at itheima.ExceptionDemo2.method(ExceptionDemo2.java:18)at itheima.ExceptionDemo2.main(ExceptionDemo2.java:12)

可以看到在输出“开始之后”,控制台在第二行输出了异常的类名ArrayIndexOutOfBoundsException及原因,第三行输出了异常出现的位置(第18行)。而程序并没有输出“结束”,这说明JVM将异常的对应信息输出之后,就将程序结束了。

综上,如果程序出现了问题,我们没有做任何处理,最终会作默认的处理:

(1)把异常的名称、异常原因以及异常出现的位置等信息输出在了控制台;

(2)程序停止执行。

1.3 异常处理

如果程序出现了问题,我们需要自己来处理,有两种方案:

(1)try……catch……

(2)throws

1.4 异常处理之try……catch……

格式:

try{可能出现异常的代码} catch(异常类名 变量名) {异常的处理代码;}

执行流程:程序从try里面的代码开始执行。出现异常,会自动生成(new)一个异常类对象,该异常对象将被提交给Java运行时系统。当Java运行时系统接收到异常对象时,会到catch中去找匹配的异常类,找到后进行异常的处理。执行完毕之后,程序还可以继续往下执行。

public class ExceptionDemo2 {public static void main(String[] args) {System.out.println("开始");method();System.out.println("结束");}public static void method() {try {int[] arr = {1, 2, 3};System.out.println(arr[3]);//new ArrayIndexOutOfBoundsException()} catch (ArrayIndexOutOfBoundsException e){//System.out.println("你访问的数组对象不存在");e.printStackTrace();}}}

1.5 Throwable的成员方法

public static void method() {try {int[] arr = {1, 2, 3};System.out.println(arr[3]);} catch (ArrayIndexOutOfBoundsException e){//ArrayIndexOutOfBoundsExceptionSystem.out.println(e.getMessage());//3System.out.println(e.toString());//java.lang.ArrayIndexOutOfBoundsException: 3}}public static void method2(){try{System.out.println(1000/0);}catch (Exception e){//ArithmeticExceptionSystem.out.println(e.getMessage());//输出/ by zeroSystem.out.println(e.toString());//java.lang.ArithmeticException: / by zero}}

getMessage():返回Throwable的详细消息字符串(视JDK版本有所不同),我们可以通过ctrl+B查看Throwable类下的getMessage(),这个方法返回了String类型的成员变量detailMessage,并在这个方法往上面翻,我们会发现很多Throwable的构造方法,这些构造方法对detailMessage进行了赋值。可知在我们new某种异常对象的时候,子类根据传入的参数(包括无参和带参)调用对应的Throwable父类参构造方法,在构造方法中对detailMessage进行赋值,我们通过getMessage()就可以得到这个值。

public class Throwable implements Serializable {private String detailMessage;略。。。public Throwable(String message) {构造方法实现过程。。。detailMessage = message;}其他构造法。。。public String getMessage() {return detailMessage;}略。。。}

toString():返回可抛出的简短描述,包括异常的原因(即getMessage()的内容)及异常的类名。

printStackTrace():把异常信息错误输出在控制台,包括异常的类名、原因及位置信息。此方法输出信息比较全,所以我们一般调用这个方法查看异常的相关信息。

1.6 编译时异常和运行时异常的区别

异常的体系结构:捕获异常(异常也是一个对象,捕获即catch)最理想的是在编译期间(javac.exe执行时),但有的错误只有在运行时(java.exe执行时)才会发生(比如:除数为0数组下表越界等)。据此,我们将异常分为编译时异常运行时异常

(1)运行时异常(也称非受检异常):指编译器不要求强制处置的异常。一般指编程时的逻辑错误,是程序员应该积极避免其出现的异常。包含java.lang.RuntimeException类及它的子类。对于这类异常,可以不作处理,因为这类异常很普遍,若全部处理可能会对程序的可读性和运行效率产生影响。(无需显示处理,也可以和编译时异常一样处理)

(2)编译时异常(也称受检异常):是指编译器要求必须处置的异常。即程序在运行时由于外界因素造成的一般性异常。编译器要求Java程序必须捕获或声明所有编译时异常。对于这类异常,如程序不处理,可能会带来意想不到的结果。(必须显示处理,否则程序就会发生错误,无法通过编译)

(3)tip:可以通过在API文档内搜索对应的异常类查看其父类及祖宗类是否有RuntimeException,有的话说明是编译时异常,否则即为运行时异常。

public class ExceptionDemo3 {public static void main(String[] args) {System.out.println("开始");method2();System.out.println("结束");}//编译时异常public static void method2() {String s = "-11-11";SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");//报红:Unhandled exception: java.text.ParseException(Unhandled未经手触过的,未经处理的)//通过查看API文档,知其父类为Exception类,说明为编译时异常//Date date = sdf.parse(s);//System.out.println(date);//解决方案try{Date date = sdf.parse(s);System.out.println(date);//Thu Nov 11 00:00:00 CST }catch (ParseException e){e.printStackTrace();}}//运行时异常public static void method() {int[] arr = {1, 2, 3};//ArrayIndexOutOfBoundsException,其爷爷类为java.lang.RuntimeException,说明为运行时异常System.out.println(arr[3]);//解决方案//try{在这里我们可以知道,编译时异常并不是一定会出现的异常,只是有可能出现,只要我们字符串s与sdf的格式是匹配的,就不会出现问题而编译器是知道会出现问题,所以报红告诉我们,让我们解决// System.out.println(arr[3]);//}catch (ArrayIndexOutOfBoundsException e){// e.printStackTrace();//}}}

1.7 异常处理之throws

虽然我们通过try…catch…可以对异常进行出路,但是并不是所有的情况我们都有权限进行异常的处理。也就是说,有些时候可能出现的异常是我们解决不了的,这个时候怎么办呢?Java对此提供了throws的处理方案格式(可以在异常报红处Alt+Enter快速生成):

throws 异常类名;

注意:这个格式是跟在方法后面的

public class ExceptionDemo4 {public static void main(String[] args) {System.out.println("开始");//method();//抛出的异常没有进行处理,现在调用的时候还是要处理try{method2();}catch (Exception e){e.printStackTrace();}System.out.println("结束");}//编译时异常//先不对异常进行处理,只是抛出(延迟)异常public static void method2() throws ParseException {String s = "-11-11";SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");Date date = sdf.parse(s);System.out.println(date);//Thu Nov 11 00:00:00 CST }//运行时异常public static void method() throws ArrayIndexOutOfBoundsException {int[] arr = {1, 2, 3};System.out.println(arr[3]);//只输出了“开始”没有输出“结束”}}

不管是运行时异常还是编译时异常,我们都可以在方法后面通过throws抛出,但是这个抛出并没有做实际的处理,真正的处理还是得通过try…catch…实现。也就是说,只要你想让程序继续往下执行,就要使用try…catch…进行处理,只不过假如我们处理不了异常,我们可以抛出去(即延迟处理),在调用的时候再进行处理(让调用者处理)。

编译时异常必须要进行处理,两种处理方案:try…catch…或者throws,如果采用throws这种方案,将来谁调用谁处理。

运行时异常可以不处理,出问题后,需要我们回来修改代码。

1.8 自定义异常

虽然Java提供了很多异常类供我们使用,但是在实际开发中,这里类并不能满足我们所有的需求。比如说我们想设置学生的考试成绩只能在0~100之间。所以就需要我们自己定义异常类来实现需求。那么我们如何让自己定义的异常类称为异常体系的一员呢?

在IDEA界面按快捷键Ctrl+N,搜索NullPointException类,发现其继承自RuntimeException类;再搜索ParseException类,发现其继承自Exception类。

也就是说如果我们自定义的类继承自RuntimeException和Exception,类就可以作为异常体系的一员。

格式:

public class 异常类名 extends Exception(){无参构造带参构造}

范例:

public class ScoreException extends Exception{public ScoreException(){}public ScoreException(String message){super(message);}}

为什么带参构造方法要将message传给父类Exception类呢?我们Ctrl+B查看super的构造方法:

public Exception(String message) {super(message);}

Ctrl+B继续跟进super

public Throwable(String message) {fillInStackTrace();detailMessage = message;}

这里的message传给了detailMessage,在上文1.5Throwable的成员方法学习中我们了解到,detailMessage是Throwable的一个成员变量,我们把message传给父类之后,detailMessage有了值,将来就可以通过getMessage()或者printStackTrace()来查看detailMessage。

public class ScoreException extends Exception{public ScoreException(){}public ScoreException(String message){super(message);}}

public class Teacher {public void check(int score) throws ScoreException {//不符合分数范围的条件,所以我们就要产生一个异常对象并抛出if(score < 0 || score>100){//抛出异常对象的关键字是throw(后面没有“s”,用于在方法体内部抛出对象)//自定义异常类的异常对象需要我们手动throw,而Java提供的异常类如果我们不抛出他也会自动new一个异常对象并throw//因为抛出的ScoreException异常继承自Exception,为编译时异常,所以check()要用throws将异常类(是类,不是对象)抛出,将来在调用check()的时候就会报红,我们就可以知道这个方法有一个编译时异常需要进行处理//当然,如果ScoreException继承自RuntimeException,可以throws也可以不throws//无参构造:不显示异常产生原因//throw new ScoreException();//带参构造方法:显示异常产生原因throw new ScoreException("你输入的分数不合法");}else{System.out.println("分数合法");}}}

public class TeacherTest {public static void main(String[] args) {Scanner sc = new Scanner(System.in);System.out.println("请输入分数:");int score = sc.nextInt();Teacher t = new Teacher();try {t.check(score);} catch (ScoreException e) {e.printStackTrace();}}}

运行结果:

throws和throw的区别:

(1)throws用在方法声明后面,跟的是异常类名;而throw用在方法体内,跟的是异常对象名;

(2)throws表示抛出异常,由该方法的调用者来处理;而throw表示抛出异常,由方法体内的语句处理;

(3)throws表示出现异常的一种可能性,并不一定会发生这些异常;而执行throw一定抛出了某种异常。

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。