Java 异常的捕获与处理详解 (一)
一,异常的产生(Exception)
异常是程序之中导致程序中断的一种指令流,异常一旦出现并且没有进行合理处理的话,那么程序就会中断执行。
An exception is a flow of instruction that causes a program to interrupt in a propram. If an exception occurs and is not properly handled, the program is interrupted.
(1)不产生异常的程序: the program without any exceptions
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
int result = 10 / 2;
System.out.println("2、除法计算结果:" + result);
System.out.println("3、除法计算结束。");
}
}
运行结果:result of operation
1、除法计算开始。
2、除法计算结果:5
3、除法计算结束。
(2)产生异常的程序, the program with an exception
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
int result = 10 / 0; // 会出现错误
System.out.println("2、除法计算结果:" + result);
System.out.println("3、除法计算结束。");
}
}
运行结果:result of operation
1、除法计算开始。Exception in thread "main"
java.lang.ArithmeticException: / by zero
at Test.main(Test.java:4)
一旦产生异常,我们发现产生异常的语句以及以后的语句将不再执行,默认情况下是进行异常信息输出,而后自动结束程序的执行。
Once an exception is generated, we find that the statement that produced the exception and the subsequent statement will no longer be executed, by default the exception information output, and then the execution of the automatic termination program.
现在我们要做是:即使异常出现了我们也要让程序正确地执行完毕。
二,异常处理
如果希望程序出现异常之后程序依然可以正常的完成的话,那么就可以使用如下的格式进行异常的处理:
try {
可能出现异常的语句 ;
} [ catch (异常类型 异常对象) {
处理异常 ;
} catch (异常类型 异常对象) {
处理异常 ;
} ... ] [finally {
不管是否出现异常,都执行此代码 ;
}]
现在,使用以上的操作处理异常处理前面除法于是出现的异常:
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
try {
int result = 10 / 0; // 异常
System.out.println("2、除法计算结果:" + result); // 之前语句有异常,此语句不再执行
} catch (ArithmeticException e) {
System.out.println(e); // 异常处理:输出错误信息,java.lang.ArithmeticException:/ by zero
}
System.out.println("3、除法计算结束。");
}
}
运行结果:
1、除法计算开始。
java.lang.ArithmeticException: / by zero
3、除法计算结束。
可以发现,加入了异常处理之后,程序中即使有了异常,程序也可以正常的执行完毕,但是异常处理时的错误输出信息和之前相比,出错的信息不明确了,那么为了让错误的信息更加的完整,一般都会调用printStackTrace()方法进行异常信息的打印,这个方法打印的异常信息是最完整的:
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
try {
int result = 10 / 0; // 异常
System.out.println("2、除法计算结果:" + result); // 之前语句有异常,此语句不再执行
} catch (ArithmeticException e) {
e.printStackTrace(); // 异常处理:输出错误信息
}
System.out.println("3、除法计算结束。");
}
}
运行结果:
1、除法计算开始。
java.lang.ArithmeticException: / by zero
at Test.main(Test.java:5)
3、除法计算结束。
此时发现,打印的信息是很完整的。
在此处就多说一点,你重复几次地去执行上面的程序的时候,你会发现以上信息输出的顺序有时候会有变动,例如下面这样
1,除法计算开始
3,除法计算结束
java.lang.ArithmeticException: / by zero
at com.nokia.test1.exception_1.main(exception_1.java:10)
个人理解是:当程序调用 e.printStackTrace(); 方法的时候,它也在继续往下执行,这时候应该是两个进程在执行着。(如有错误欢迎指正!!!)
除了try…catch格式处理异常外,还可以使用try…catch..finally:
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
try {
int result = 10 / 1;
System.out.println("2、除法计算结果:" + result);
} catch (ArithmeticException e) {
e.printStackTrace();
} finally {
System.out.println("不管是否出现异常都执行");
}
System.out.println("3、除法计算结束。");
}
}
运行结果:
1、除法计算开始。
2、除法计算结果:10
不管是否出现异常都执行
3、除法计算结束。
但是,对于之前的程序又有了问题:现在执行数学计算的两个参数,都是由程序默认提供,那么如果说现在两个计算的参数通过初始化参数传递呢?
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
try {
int x = Integer.parseInt(args[0]); // 接收参数
int y = Integer.parseInt(args[1]); // 接收参数
int result = x / y;
System.out.println("2、除法计算结果:" + result);
} catch (ArithmeticException e) {
e.printStackTrace();
} finally {
System.out.println("不管是否出现异常都执行");
}
System.out.println("3、除法计算结束。");
}
}
这个时候,数据由外部传送,那么就有可能出现以下几类问题:
(1)执行时不输入参数(java TestDemo),ArrayIndexOutOfBoundsException,未处理;
(2)输入的参数不是数字(java TestDemo a b),NumberFormatException,未处理;
(3)被除数为0(java TestDemo 10 0),ArithmeticException,已处理。
可以发现,以上的程序实际上是存在三种异常,而程序之中只能够处理一种,而对于不能处理的异常,发现程序依然会直接中断执行。
加入多个catch:
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
try {
int x = Integer.parseInt(args[0]);
int y = Integer.parseInt(args[1]);
int result = x / y;
System.out.println("2、除法计算结果:" + result);
} catch (ArithmeticException e) {
e.printStackTrace();
} catch (ArrayIndexOutOfBoundsException e) {
e.printStackTrace();
} catch (NumberFormatException e) {
e.printStackTrace();
} finally {
System.out.println("不管是否出现异常都执行");
}
System.out.println("3、除法计算结束。");
}
}
现在,程序比之前更健壮了。
三,异常处理流程
以上已经完成了异常的基本处理,但是所有的异常都想之前那样一条条判断似乎是不可能完成的一件事,因为日后肯定会接触到一些不常见的异常信息,那么下面就首先研究一下JVM异常的处理流程和结构。
先查看两个异常类的继承结构:
(1)ArithmeticException:
java.lang.Object
|- java.lang.Throwable
|- java.lang.Exception
|- java.lang.RuntimeException
|- java.lang.ArithmeticException
(2)ArrayIndexOutOfBoundsException:
java.lang.Object
|- java.lang.Throwable
|- java.lang.Exception
|- java.lang.RuntimeException
|- java.lang.IndexOutOfBoundsException
|-java.lang.ArrayIndexOutOfBoundsException
可以发现,所有的异常类型最高的继承类是Throwable,Throwable下有两个子类:
(1)Error:指的是JVM错误,这个时候的程序并没有执行,无法处理;
(2)Exception:指的是程序之中出现的错误信息,可以进行异常处理。
通过继承关系可以发现,在进行日后异常处理的时候是以Exception为主,并且可以形成以下的异常处理流程:
(1)如果程序中产生了异常,那么JVM根据异常的类型,实例化一个指定异常类的对象;
(2)如果这时程序中没有任何的异常处理操作,则这个异常类的实例化对象将交给JVM进行处理,而JVM的默认处理方式就是进行异常信息的输出,而后中断程序执行;
(3)如果程序中存在了异常处理,则会由try语句捕获产生的异常类对象;
(4)与try之后的每一个catch进行匹配,如果匹配成功,则使用指定的catch进行处理,如果没有匹配成功,则向后面的catch继续匹配,如果没有任何的catch匹配成功,则这个时候将交给JVM执行默认处理;
(5)不管是否有异常都会执行finally程序,如果此时没有异常,执行完finally,则会继续执行程序之中的其他代码,如果此时有异常没有能够处理(没有一个catch可以满足),那么也会执行finally,但是执行完finally之后,将默认交给JVM进行异常的信息输出,并且程序中断。
通过以上的分析可以发现,实际上catch捕获异常类型的操作,就和方法接收参数是一样的,那么按照之前所学习过的对象多态性来讲,所有的异常类都是Exception的子类,那么这个时候,实际上所有的异常都可以使用Exception进行接收:
public class Test {
public static void main(String args[]) {
System.out.println("1、除法计算开始。");
try {
int x = Integer.parseInt(args[0]);
int y = Integer.parseInt(args[1]);
int result = x / y;
System.out.println("2、除法计算结果:" + result);
} catch (Exception e) {
e.printStackTrace();
} finally {
System.out.println("不管是否出现异常都执行");
}
System.out.println("3、除法计算结束。");
}
}
这时应该可以感受到异常处理所带来的好处了。但是这种操作也存在一种问题:如果在一些异常处理要求严格的项目之中,异常必须分别处理,如果现在异常的处理要求不是很严格,直接编写Exception就足够了。
未完待续。。。
Java 异常的捕获与处理详解 (一)的更多相关文章
- Java 异常的捕获与处理详解(二)
(一).throws关键字 throws关键字主要是在定义上使用的,表示的是此方法中不进行异常处理,而交给被调用处处理. 例如: class MyMath { public int div(int x ...
- “全栈2019”Java异常第二十章:自定义异常详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java异 ...
- Kotlin异常与Java异常的区别及注解详解
Kotlin异常与Java异常的区别: throw的Kotlin中是个表达式,这样我们可以将throw作为Elvis表达式[val test = aa ?: bb,这样的则为Elvis表达式,表示如果 ...
- Mysql高手系列 - 第20篇:异常捕获及处理详解(实战经验)
Mysql系列的目标是:通过这个系列从入门到全面掌握一个高级开发所需要的全部技能. 这是Mysql系列第20篇. 环境:mysql5.7.25,cmd命令中进行演示. 代码中被[]包含的表示可选,|符 ...
- java的集合框架最全详解
java的集合框架最全详解(图) 前言:数据结构对程序设计有着深远的影响,在面向过程的C语言中,数据库结构用struct来描述,而在面向对象的编程中,数据结构是用类来描述的,并且包含有对该数据结构操作 ...
- 牛客网 Java 工程师能力评估 20 题 - 详解
牛客网 Java 工程师能力评估 20 题 - 详解 不知在看博客的你是否知道 牛客网,不知道就太落后了,分享给你 : 牛客网 此 20 题,绝对不只是 20 题! 免责声明:本博客为学习笔记,如有侵 ...
- 【转】Java魔法堂:String.format详解
Java魔法堂:String.format详解 目录 一.前言 二.重载方法 三.占位符 四.对字符.字符串进行格式化 五.对整数进行格式化 六. ...
- java线程池的使用与详解
java线程池的使用与详解 [转载]本文转载自两篇博文: 1.Java并发编程:线程池的使用:http://www.cnblogs.com/dolphin0520/p/3932921.html ...
- JAVA通过JDBC连接Oracle数据库详解【转载】
JAVA通过JDBC连接Oracle数据库详解 (2011-03-15 00:10:03) 转载▼http://blog.sina.com.cn/s/blog_61da86dd0100q27w.htm ...
随机推荐
- 51nod 1717 好数 (水题)
题目: 看起来很复杂,其实就是有多少个素因子就翻转多少次. 然后考虑到只有平方数有奇数个素因子. 一次过,上代码把: #include <iostream> #include <al ...
- jQuery学习(二)——使用JQ完成页面定时弹出广告
1.JQuery效果 2.步骤分析: 第一步:引入jQuery相关的文件 第二步:书写页面加载函数 第三步:在页面加载函数中,获取显示广告图片的元素. 第四步:设置定时操作(显示广告图片的函数) 第五 ...
- QT笔记 -- (3) 为QLabel添加鼠标响应方法1
参考 http://qt-project.org/wiki/Make-a-QLabel-Clickable 1.首先重载QLabel的mousePressEvent,这样点击QLabel时就能发出cl ...
- swift 20 - Nested Types
Nested Types 只是为了方便类型的整合和使用 struct BlackjackCard { // nested Suit enumeration enum Suit: Character { ...
- Github上值得关注的前端项目-转自好友trigkit4
http://microjs.com/# 该网站的资源都托管到了github,microjs.com是一个可以让你选择微型的js类库的网站,该网站里的js库都是压缩后不大于5KB的,非常实用 http ...
- WordPress开启伪静态
一.NGINX 的话在 domain.conf 的 server 增加代码: location / { try_files $uri $uri/ /index.php?$args; } 如果使用的是 ...
- Vue 中 换行符获取
当要获取到 vue 中 文本域的换行符时, 需要用到正则匹配. let reg = new RegExp('/n',"g"); let str = text.replace(reg ...
- mysql 密码的破解
现在的主流的数据库一般是mysql ,sql server , oracle. 有的时候我们忘记了数据库密码的时候我们要怎么办,破解别人的数据库的密码的时候我们要怎么搞 忘记密码是一件很头痛的 ...
- c++ 子类构造函数初始化及父类构造初始化
我们知道,构造方法是用来初始化类对象的.如果在类中没有显式地声明构造函数,那么编译器会自动创建一个默认的构造函数:并且这个默认的构造函数仅仅在没有显式地声明构造函数的情况下才会被创建创建. 构造函数与 ...
- docker 下修改 mysql sql_mode和配置文件
原文:docker 下修改 mysql sql_mode和配置文件 打开PowerShell 首先创建mysql容器,这里我们指定使用mysql5.7的版本 docker run -d -p 3306 ...