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

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

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

  • 在开发过程中,即是我们把代码写得尽善尽美,在系统运行过程中仍然会遇到一些问题,因为很多问题不是靠代码能够避免的,如:客户输入数据的格式、读取文件是否存在、网络是否始终保持通畅等。
  • 异常:在Java语言中,将程序执行中发生的不正常情况称为“异常”。(开发过程中的语法错误和逻辑错误等代码问题不是异常)
  • Java程序执行时发生的异常事件的分类:
    (1)Error:Java虚拟机无法解决的严重问题(程序执行分为编译和运行两个过程,运行的时候特别需要使用JVM)。如:JVM系统内部错误、资源耗尽(比如StackOverflowError(栈溢出,如下例:递归调用导致的栈溢出)和OOM(堆溢出))等严重情况。一般不(无法)编写针对性的代码进行处理。
    (2)Exception(狭义上的异常,我们所说的异常处理通常就是指这个,不包括Error):其他因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。如:
    (1)空指针访问
    (2)试图读取不存在的文件
    (3)网络连接中断
    (4)数组角标越界
//资源耗尽的两种情况
//1、递归调用导致的栈溢出java.lang.StackOverflowError
public class ErrorTest {public static void main(String[] args) {main(args);}
}//2、new的空间过大,堆溢出java.lang.OutOfMemoryError
public 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 String getMessage() 返回Throwable的详细消息字符串
public String toString() 返回可抛出的简短描述
public void printStackTrace() 把异常信息错误输出在控制台
    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 = "2021-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 2021}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 = "2021-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 2021}//运行时异常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一定抛出了某种异常。

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

  1. 黑马程序员:Java基础总结----Java语言编程规范

       黑马程序员:Java基础总结        Java语言编程规范:参考自SUN公司文档  ASP.Net+Android+IO开发..Net培训.期待与您交流!  I.   排版规范 A.  规 ...

  2. 黑马程序员入学Java知识——精华总结

    黑马程序员入学Java知识--精华总结 J2SE部分,Java高新技术部分,7K面试题部分等黑马入学要求的知识点总结! 一.黑马程序员-java概述与基础知识 6 1.何为编程? 6 2.Java语言 ...

  3. 2016最新整理黑马程序员JavaEE第28期 基础班就业班

    2016最新整理黑马程序员JavaEE第28期 基础班就业班 http://blog.sina.com.cn/s/blog_88d972910102wti3.html

  4. 《黑马程序员2023新版黑马程序员大数据入门到实战教程,大数据开发必会的Hadoop、Hive,云平台实战项目》学习笔记总目录

    本文是对<黑马程序员新版大数据入门到实战教程>所有知识点的笔记进行总结分类. 学习视频:黑马程序员新版大数据 学习时总结的学习笔记以及思维导图会在后续更新,请敬请期待. 前言:配置三台虚拟 ...

  5. 黑马程序员C++ 第一阶段 C++基础语法入门

    B站黑马C++内容,自己手动敲了一遍代码,文章之间也加入了一些我个人的理解,仅供学习和参考用,程序代码均来自黑马程序员 . 一.C++基础入门 1.1第一个c++程序 (输出一个hello world ...

  6. 黑马程序员C++学习笔记<第一阶段_基础篇>

    配套视频网址: 黑马程序员:http://yun.itheima.com/course/520.html?bili B站:https://www.bilibili.com/video/BV1et411 ...

  7. idea java 单元测试_Java基础教程:IDEA单元测试

    Java基础教程:IDEA单元测试 环境配置 使用idea IDE 进行单元测试,首先需要安装JUnit 插件. 安装JUnit插件步骤 File-->settings-->Plguins ...

  8. 黑马程序员之Java集合类详解

    android培训  java培训 期待与您交流! Collection接口 Collection是最基本的集合接口,一个Collection代表一组Object,即Collection的元素(Ele ...

  9. 黑马程序员:java基础——OOP面向对象的特征

    ----------- android培训.java培训.java学习型技术博客.期待与您交流! ------------   一概述: 一.面向对象(OOP)的概述: 1)简介:OOP: Objec ...

最新文章

  1. 局部变量 final Java_Java局部变量final
  2. php mysql购物车_php mysql购物车实现程序
  3. Jmeter性能测试之if控制器的使用
  4. 写一副对子_挥毫泼墨写春联 西安街头年味分外浓
  5. UTF-8格式的文本文件程序读取异常
  6. 我三年开发经验,从字节跳动抖音离职后,吐血整理
  7. python 概率分布类型检验_统计学:假设检验Python案例实现+概率论基础知识回顾...
  8. linux下,如何迁移mysql数据库存放目录。利用软连接简单快捷实现移动到home数据盘下...
  9. 写shell工具类,一个常用实例
  10. 【观察】打造产业数字引擎背后,紫光云价值使命的新跃迁
  11. Android Studio开发记录
  12. bind mysql web_bindview+dlz(mysql)
  13. 音视频处理基础知识扫盲:数字视频YUV像素表示法以及视频帧和编解码概念介绍
  14. 职场神攻略:5分钟自我介绍法 快准狠!
  15. android 播放提示音,[转载]android播放音效例子 (翻页音效、警报音效通用
  16. C++面试题整理(二)
  17. 微信小程序商城:解决用户拉新与留存问题
  18. Levy distribution(列维分布)和Levy fligt(列维飞行)
  19. Linux系统ORACLE 19C OEM监控管理
  20. 实现员工考勤信息管理

热门文章

  1. 这些段子只有程序员才能看懂?
  2. express的学习
  3. 区块链的核心:共识机制
  4. 小球落地java_JAVA------------------华为机试----------------求小球落地5次后所经历的路程和第5次反弹的高度...
  5. 孔子学院年度报告(2006-2018)
  6. C#如何连接SAP调用SAP接口函数
  7. jquery动态获取id
  8. 【HTML】一款不错的前端开发IDE——Aptana Studio
  9. 2016年7月微软MVP申请开始了!
  10. 马云说了:年轻人,你不去创业,不去 旅游