一,异常的产生(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 异常的捕获与处理详解 (一)的更多相关文章

  1. Java 异常的捕获与处理详解(二)

    (一).throws关键字 throws关键字主要是在定义上使用的,表示的是此方法中不进行异常处理,而交给被调用处处理. 例如: class MyMath { public int div(int x ...

  2. “全栈2019”Java异常第二十章:自定义异常详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java异 ...

  3. Kotlin异常与Java异常的区别及注解详解

    Kotlin异常与Java异常的区别: throw的Kotlin中是个表达式,这样我们可以将throw作为Elvis表达式[val test = aa ?: bb,这样的则为Elvis表达式,表示如果 ...

  4. Mysql高手系列 - 第20篇:异常捕获及处理详解(实战经验)

    Mysql系列的目标是:通过这个系列从入门到全面掌握一个高级开发所需要的全部技能. 这是Mysql系列第20篇. 环境:mysql5.7.25,cmd命令中进行演示. 代码中被[]包含的表示可选,|符 ...

  5. java的集合框架最全详解

    java的集合框架最全详解(图) 前言:数据结构对程序设计有着深远的影响,在面向过程的C语言中,数据库结构用struct来描述,而在面向对象的编程中,数据结构是用类来描述的,并且包含有对该数据结构操作 ...

  6. 牛客网 Java 工程师能力评估 20 题 - 详解

    牛客网 Java 工程师能力评估 20 题 - 详解 不知在看博客的你是否知道 牛客网,不知道就太落后了,分享给你 : 牛客网 此 20 题,绝对不只是 20 题! 免责声明:本博客为学习笔记,如有侵 ...

  7. 【转】Java魔法堂:String.format详解

    Java魔法堂:String.format详解     目录     一.前言    二.重载方法     三.占位符     四.对字符.字符串进行格式化     五.对整数进行格式化     六. ...

  8. java线程池的使用与详解

    java线程池的使用与详解 [转载]本文转载自两篇博文:  1.Java并发编程:线程池的使用:http://www.cnblogs.com/dolphin0520/p/3932921.html   ...

  9. JAVA通过JDBC连接Oracle数据库详解【转载】

    JAVA通过JDBC连接Oracle数据库详解 (2011-03-15 00:10:03) 转载▼http://blog.sina.com.cn/s/blog_61da86dd0100q27w.htm ...

随机推荐

  1. 001.ActiveMQ概述

    1. 概念 ActiveMQ是Apache推出的,一款开源的,完全支持JMS1.1和J2EE1.4规范的JMS Provider实现的消息中间件(Message Oriented Middleware ...

  2. MySQL学习(五)——使用JDBC完成用户表CRUD的操作

    通过案例我们发现“获得连接”和“释放资源”两次代码将在之后的增删改查所有功能中都存在,开发中遇到此种情况,将采用工具类的方法进行抽取,从而达到代码的重复利用. 1.使用properties配置文件 开 ...

  3. <ItemTemp>里写判断语句

    <%@ Language="C#" %> <html> <head></head> <body> <%=DateT ...

  4. sass的用法小结(二)

    3. 导入SASS文件; css有一个特别不常用的特性,即@import规则,它允许在一个css文件中导入其他css文件.然而,后果是只有执行到@import时,浏览器才会去下载其他css文件,这导致 ...

  5. (noip模拟十七)【BZOJ3930】[CQOI2015]选数-容斥水法

    Description 我们知道,从区间[L,H](L和H为整数)中选取N个整数,总共有(H-L+1)^N种方案.小z很好奇这样选出的数的最大公约数的规律,他决定对每种方案选出的N个整数都求一次最大公 ...

  6. Markdown语法简记

    目录 一.标题 1. 六个级别的标题 2. 主.副两级标题 二.根据标题生成文档结构大纲 三.字体 1. 斜体 2. 粗体 3. 倾斜加粗 4. 行首缩进 5. 删除线 四.引用块 五.代码块 1. ...

  7. Java基础学习总结(43)——Java8 Lambda揭秘

    再了解了Java 8 Lambda的一些基本概念和应用后, 我们会有这样的一个问题: Lambda表达式被编译成了什么?. 这是一个有趣的问题,涉及到JDK的具体的实现. 本文将介绍OpenJDK对L ...

  8. 利用hive源码解析sql查了哪些表哪些字段

    在hiveserver2中使用了org.apache.hadoop.hive.ql.parse.BaseSemanticAnalyzer对抽象语法树(AST)进行语义分析,分析的过程可以得出hive查 ...

  9. 调用Windows属性窗口

    简述 在Windows系统下.可以通过:右键 -> 属性,来查看文件/文件夹对应的属性信息,包括:常规.安全.详细信息等. 简述 共有类型 共有类型 首先,需要包含头文件: #include & ...

  10. C - The C Answer (2nd Edition) - Exercise 1-12

    /* Write a program that prints its input one word per line. */ #include <stdio.h> #define IN 1 ...