黑马程序员全套Java教程_Java基础教程_异常(含扩展)(二十三)
黑马程序员全套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基础教程_异常(含扩展)(二十三)相关推荐
- 黑马程序员:Java基础总结----Java语言编程规范
黑马程序员:Java基础总结 Java语言编程规范:参考自SUN公司文档 ASP.Net+Android+IO开发..Net培训.期待与您交流! I. 排版规范 A. 规 ...
- 黑马程序员入学Java知识——精华总结
黑马程序员入学Java知识--精华总结 J2SE部分,Java高新技术部分,7K面试题部分等黑马入学要求的知识点总结! 一.黑马程序员-java概述与基础知识 6 1.何为编程? 6 2.Java语言 ...
- 2016最新整理黑马程序员JavaEE第28期 基础班就业班
2016最新整理黑马程序员JavaEE第28期 基础班就业班 http://blog.sina.com.cn/s/blog_88d972910102wti3.html
- 《黑马程序员2023新版黑马程序员大数据入门到实战教程,大数据开发必会的Hadoop、Hive,云平台实战项目》学习笔记总目录
本文是对<黑马程序员新版大数据入门到实战教程>所有知识点的笔记进行总结分类. 学习视频:黑马程序员新版大数据 学习时总结的学习笔记以及思维导图会在后续更新,请敬请期待. 前言:配置三台虚拟 ...
- 黑马程序员C++ 第一阶段 C++基础语法入门
B站黑马C++内容,自己手动敲了一遍代码,文章之间也加入了一些我个人的理解,仅供学习和参考用,程序代码均来自黑马程序员 . 一.C++基础入门 1.1第一个c++程序 (输出一个hello world ...
- 黑马程序员C++学习笔记<第一阶段_基础篇>
配套视频网址: 黑马程序员:http://yun.itheima.com/course/520.html?bili B站:https://www.bilibili.com/video/BV1et411 ...
- idea java 单元测试_Java基础教程:IDEA单元测试
Java基础教程:IDEA单元测试 环境配置 使用idea IDE 进行单元测试,首先需要安装JUnit 插件. 安装JUnit插件步骤 File-->settings-->Plguins ...
- 黑马程序员之Java集合类详解
android培训 java培训 期待与您交流! Collection接口 Collection是最基本的集合接口,一个Collection代表一组Object,即Collection的元素(Ele ...
- 黑马程序员:java基础——OOP面向对象的特征
----------- android培训.java培训.java学习型技术博客.期待与您交流! ------------ 一概述: 一.面向对象(OOP)的概述: 1)简介:OOP: Objec ...
最新文章
- 局部变量 final Java_Java局部变量final
- php mysql购物车_php mysql购物车实现程序
- Jmeter性能测试之if控制器的使用
- 写一副对子_挥毫泼墨写春联 西安街头年味分外浓
- UTF-8格式的文本文件程序读取异常
- 我三年开发经验,从字节跳动抖音离职后,吐血整理
- python 概率分布类型检验_统计学:假设检验Python案例实现+概率论基础知识回顾...
- linux下,如何迁移mysql数据库存放目录。利用软连接简单快捷实现移动到home数据盘下...
- 写shell工具类,一个常用实例
- 【观察】打造产业数字引擎背后,紫光云价值使命的新跃迁
- Android Studio开发记录
- bind mysql web_bindview+dlz(mysql)
- 音视频处理基础知识扫盲:数字视频YUV像素表示法以及视频帧和编解码概念介绍
- 职场神攻略:5分钟自我介绍法 快准狠!
- android 播放提示音,[转载]android播放音效例子 (翻页音效、警报音效通用
- C++面试题整理(二)
- 微信小程序商城:解决用户拉新与留存问题
- Levy distribution(列维分布)和Levy fligt(列维飞行)
- Linux系统ORACLE 19C OEM监控管理
- 实现员工考勤信息管理