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 ...
随机推荐
- 【参考】Linux下的Memcache安装
服务器端主要是安装memcache服务器端.下载:http://www.danga.com/memcached/dist/memcached-1.2.2.tar.gz另外,Memcache用到了lib ...
- swift学习笔记 - swift中常用关键字
swift中常用关键字 **用作声明的关键字: ** class.deinit.enum.extension.func.import.init.let.protocol.static.struct.s ...
- 3ds Max绘制青花瓷茶壶
1.在桌面找到3DMAX软件,左键双击,启动程序: 2.在命令案板中,找到几何体,茶壶,在顶视图绘制一个茶壶: 3.在百度图片中搜索,查找“青花瓷”,找到一个自己喜欢的精美图案,截图保存备用: 4.在 ...
- ZBrush实用插件ZAppLink简介
ZAppLink是ZBrush版本推出时被评为最值得期待的插件.事实证明,ZAppLink的出现让工具与工具之间有了交流,搭起软件与软件的沟通桥梁. ZAppLink插件专用于扩展ZBrush®的绘制 ...
- HDU-4296 Buildings 贪心 从相邻元素的相对位置开始考虑
题目链接:https://cn.vjudge.net/problem/HDU-4296 题意 有很多板子,每一个板子有重量(w)和承重(s)能力 现规定一块板子的PDV值为其上所有板子的重量和减去这个 ...
- C++ STL rope介绍----可持久化平衡树
大致介绍: rope这个东西,我刚刚知道这玩意,用的不是很多,做个简单的介绍. 官方说明:我是刘邦(我估计你是看不懂的). rope就是一个用可持久化平衡树实现的“重型”string(然而它也可以保存 ...
- LOJ #10121 与众不同 (RMQ+二分)
题目大意 :给你一个整数序列,定义一个合法子串为子串内所有数互不相同,会有很多询问,求区间$[L,R]$内最长连续合法子串长度 一道思维不错的$RMQ$题,NOIP要是考这种题可能会考挂一片 预处理出 ...
- JavaScript系列——数组元素左右移动N位算法实现
引言 在自己刚刚毕业不久的时候,去了一家公司面试,面试官现场考了我这道题,我记忆深刻,当时没有想到思路,毫无疑问被面试官当成菜鸟了.最近刚好在研究数组的各种算法实现,就想到这道题,可以拿来实现一下,纪 ...
- HDU 1039.Easier Done Than Said?【字符串处理】【8月24】
Easier Done Than Said? Problem Description Password security is a tricky thing. Users prefer simple ...
- lenovo G系列重装系统
lenovo G41 的笔记本默认安装的是win8 中文版 的操作系统,使用非常不方便,用U盘重装成WIN7的系统. 1.用启动工具软件制作U盘启动盘. 详细能够參照 http://www.uqi ...