Java 异常(一) 异常概述及其架构
Java 异常(一) 异常概述及其架构
一、异常概述
(一)、概述
Java异常是Java提供的一种识别及响应错误的一致性机制。异常指的是程序在执行过程中,出现的非正常的情况,最终会导致JVM的非正常停止。
Java异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程序健壮性。在有效使用异常的情况下,异常能清晰的回答 what, where, why 这3个问题:异常类型回答了“什么”被抛出,异常堆栈跟踪回答了“在哪“抛出,异常信息回答了“为什么“会抛出。
(二)、异常体系
异常机制其实是帮助我们找到程序中的问题,异常的根类是java.lang.Throwable,其下有两个子类:java.lang.Error与java.lang.Exception,平常所说的异常指java.lang.Exception。
Throwable: 有两个重要的子类:Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。异常和错误的区别是:异常能被程序本身可以处理,错误是无法处理。
1、Throwable:Throwable是 Java 语言中所有错误或异常的超类。Throwable包含两个子类: Error 和 Exception。它们通常用于指示发生了异常情况。
Throwable包含了其线程创建时线程执行堆栈的快照,它提供了printStackTrace()等接口用于获取堆栈跟踪数据等信息。
Throwable常用API:
public void printStackTrace() // 打印异常的详细信息。包含了异常的类型,异常的原因,还包括异常出现的位置。
public String getMessage() // 获取发生异常的原因。
public String toString() // 获取异常的类型和异常描述信息。
2、Error:严重错误Error,无法通过处理的错误,只能事先避免。
3、Exception:表示异常,异常产生后程序员可以通过代码的方式纠正,使程序继续运行,是必须要处理的。
(三)、异常分类
我们平常说的异常就是指Exception,因为这类异常一旦出现,我们就要对代码进行更正,修复程序。
异常主要分为两大类:编译期异常和运行期异常。
编译期异常:checked异常。在编译期,就会检查,如果没有处理异常,则编译失败(如日期格式化异常)。
特点: Java编译器会检查它。此类异常,要么通过throws进行声明抛出,要么通过try-catch进行捕获处理,否则不能通过编译。例如,CloneNotSupportedException就属于被检查异常。当通过clone()接口去克隆一个对象,而该对象对应的类没有实现Cloneable接口,就会抛出CloneNotSupportedException异常。被检查异常通常都是可以恢复的。
运行期异常:runtime异常。在运行时期,检查异常.在编译时期,运行异常不会编译器检测(不报错)(如数学异常)。
特点: Java编译器不会检查它。也就是说,当程序中可能出现这类异常时,倘若既"没有通过throws声明抛出它",也"没有用try-catch语句捕获它",还是会编译通过。例如,除数为零时产生的ArithmeticException异常,数组越界时产生的IndexOutOfBoundsException异常,fail-fail机制产生的ConcurrentModificationException异常等,都属于运行时异常。
虽然Java编译器不会检查运行时异常,但是我们也可以通过throws进行声明抛出,也可以通过try-catch对它进行捕获处理。
如果产生运行时异常,则需要通过修改代码来进行避免。例如,若会发生除数为零的情况,则需要通过代码避免该情况的发生!
二、异常处理
Java异常处理机制用到的几个关键字:try、catch、finally、throw、throws。
try:用于监听。将要被监听的代码(可能抛出异常的代码)放在try语句块之内,当try语句块内发生异常时,异常就被抛出。
catch:用于捕获异常。catch用来捕获try语句块中发生的异常。
finally:finally语句块总是会被执行。它主要用于回收在try块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。只有finally块,执行完成之后,才会回来执行try或者catch块中的return或者throw语句,如果finally中使用了return或者throw等终止方法的语句,则就不会跳回执行,直接停止。
throw:用于抛出异常。
throws:用在方法签名中,用于声明该方法可能抛出的异常。
三、异常注意点
(一)、finally中的异常会覆盖(消灭)前面try或者catch中的异常
不要在fianlly中使用return。
不要在finally中抛出异常。
减轻finally的任务,不要在finally中做一些其它的事情,finally块仅仅用来释放资源是最合适的。
将尽量将所有的return写在函数的最后面,而不是try ... catch ... finally中。
(二)、多个异常使用捕获
多个异常分别处理。
多个异常一次捕获,多次处理。
多个异常一次捕获一次处理。
一般情况下,一般我们是使用一次捕获多次处理方式,如下代码
try{
// 编写可能会出现异常的代码
}catch(异常类型A e){ 当try中出现A类型异常,就用该catch来捕获.
// 处理异常的代码 //记录日志/打印异常信息/继续抛出异常
}catch(异常类型B e){ 当try中出现B类型异常,就用该catch来捕获.
// 处理异常的代码 //记录日志/打印异常信息/继续抛出异常
}
注意:这种异常处理方式,要求多个catch中的异常不能相同,并且若catch中的多个异常之间有子父类异常的关系,那么子类异常要求在上面的catch处理,父类异常在下面的catch处理。
三、实例
(一)、try - catch用法
try - catch 必须搭配使用,不能单独使用。
public class ExceptionDemo {
public static void main(String[] args) { try { int i = 10 / 0;// 除数不能为0,此行会抛出 ArithmeticException 异常 System.out.println("i = " + i);// 抛出异常后,此行不会执行 }catch(ArithmeticException e) { // 捕获 ArithmeticException 异常 // 在catch 代码块处理异常 e.printStackTrace(); // 异常最详细信息 System.out.println("e.getMessage() : " + e.getMessage());// 发生异常的原因 System.out.println("e.toString() : " + e.toString()); // 获取异常的类型和异常描述信息 } }
}
(二)、finally 用法
try - catch - finally搭配使用,或者 try - finally 搭配使用。
public class ExceptionDemo {
public static void main(String[] args) { // try-catch-finally搭配使用 try { int[] arr = {1,2,3}; int i = arr[3];// 数组索引越界,此行会抛出 ArrayIndexOutOfBoundsException 异常 System.out.println("i = " + i);// 抛出异常后,此行不会执行 }catch(ArithmeticException e) { // 捕获 ArithmeticException System.out.println(e.getMessage());// 发生异常的原因 System.exit(0); // 程序强制退出,finally 代码块不会执行 }finally {// 除了程序强制退出,如(System。exit(0)),无论是否发生异常,finally 代码块总会执行 System.out.println("this is finally"); } // try-finally搭配使用 try { int[] arr = {1,2,3}; int i = arr[3];// 数组索引越界,此行会抛出 ArrayIndexOutOfBoundsException 异常 System.out.println("i = " + i);// 抛出异常后,此行不会执行 }finally { // 无论是否发生异常,finally 代码块总会执行 System.out.println("this is finally"); } }
}
注意点:
try-catch-finally 搭配:这种形式捕获异常时,开发者可以在 catch 代码块中处理异常(如打印日志、日志记录等),异常处理权在开发者。
try-finally 搭配:这种形式捕获异常时,默认抛出给 JVM 处理,JVM默认处理时调用 e.printStackTrace() 方法打印异常详细信息。
finally 代码块:除非程序强制退出,否则无论程序是否发生异常,finally 代码块总会执行。
finally 中抛出异常会覆盖(消灭)前面 try 或者 catch 中的异常,尽量避免在 finally 代码块中抛出异常。
如果 try 中和 finally 中都有 return 语句,程序会先执行 finally 中的 return 语句,然后程序块运行结束,而不会执行 try 中的 return 语句。所以尽量不在finally中使用 return 语句。
(三)、throw 用法
throw 是用于抛出异常,将这个异常对象传递到调用者处,并结束当前方法的执行
public static void main(String[] args) {
try {
int i = 10 / 0;
System.out.println("i = " + i);
}catch(ArithmeticException e) {
// 抛出异常,传递自定义异常信息提示
// 默认抛出给 JVM 处理打印异常详细信息
throw new ArithmeticException("除数不能为0");
}
}
(四)、throws 用法
throws运用于方法声明之上,用于表示当前方法不处理异常,而是提醒该方法的调用者来处理异常(抛出异常)。
public class ExceptionDemo {
public static void main(String[] args) { demo(); } public static void demo() throws ArrayIndexOutOfBoundsException{ try { int[] arr = {1,2,3}; int i = arr[3]; System.out.println("i = " + i); }catch(ArrayIndexOutOfBoundsException e) { System.out.println(e.toString()); } }
}
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Kafka的生产者优秀架构设计
Kafka 是一个高吞吐量的分布式的发布订阅消息系统,在全世界都很流行,在大数据项目里面使用尤其频繁。笔者看过多个大数据开源产品的源码,感觉 Kafka 的源码是其中质量比较上乘的一个,这得益于作者高超的编码水平和高超的架构设计能力。 Kafka 的核心源码分为两部分:客户端源码和服务端源码,客户端又分为生产者和消费者,而个人认为 Kafka 的源码里面生产者的源码技术含量最高,所以今天给大家剖析 Kafka 的生产者的架构设计,Kafka 是一个飞速发展的消息系统,其架构也在一直演进中,我们今天分析的 Kafka 的版本是比较成熟稳定的 Kafka1.0.0 版本源码。图1 Kafka核心模块生产者流程概述 先给大家介绍一下生产者的大概的运行的流程。图2 Kafka运行方式 如上图所示:步骤一:一条消息过来首先会被封装成为一个 ProducerRecord 对象。 步骤二:接下来要对这个对象进行序列化,因为 Kafka 的消息需要从客户端传到服务端,涉及到网络传输,所以需要实现序列。Kafka 提供了默认的序列化机制,也支持自定义序列化(这种设计也值得我们积累,提高项目的扩展性)。 ...
- 下一篇
物联网边缘计算产品架构三大设备
云栖号快速入门:【点击查看更多云产品快速入门】不知道怎么入门?这里分分钟解决新手入门等基础问题,可快速完成产品配置操作! 本章主要介绍物联网边缘计算的产品架构。 产品架构如下图所示。物联网边缘计算主要涉及设备端、边缘计算端和云端三个部分: 设备端开发者使用设备接入SDK,将非标设备转换成标准物模型,就近接入网关,从而实现设备的管理和控制。 边缘计算端设备连接到网关后,网关可以实现设备数据的采集、流转、存储、分析和上报设备数据至云端,同时网关提供规则引擎、函数计算引擎,方便场景编排和业务扩展。 云端设备数据上传云端后,可以结合阿里云功能,如大数据、AI学习等,通过标准API接口,实现更多功能和应用。 本文来自 阿里云文档中心 物联网边缘计算 产品架构 【云栖号在线课堂】每天都有产品技术专家分享!课程地址:https://yqh.aliyun.com/zhibo 立即加入社群,与专家面对面,及时了解课程最新动态!【云栖号在线课堂 社群】https://c.tb.cn/F3.Z8gvnK
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS7安装Docker,走上虚拟化容器引擎之路
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- CentOS7编译安装Gcc9.2.0,解决mysql等软件编译问题
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- Docker安装Oracle12C,快速搭建Oracle学习环境
- CentOS7设置SWAP分区,小内存服务器的救世主
- SpringBoot2全家桶,快速入门学习开发网站教程
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- CentOS8安装Docker,最新的服务器搭配容器使用
- CentOS关闭SELinux安全模块