抱歉,您的浏览器无法访问本站
本页面需要浏览器支持(启用)JavaScript
了解详情 >

15 异常处理

异常定义

导致程序的正常流程被中断的事件,叫做异常

异常处理常见手段:try catch finally throws

例如:文件不存在异常

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
package ExceptionProcess;

import java.io.File;
import java.io.FileInputStream;

public class Test {

public static void main(String[] args){
File f = new File("D:/LOL.exe");
new FileInputStream(f);
}

}


/*输出:
Exception in thread "main" java.lang.Error: Unresolved compilation problem:
Unhandled exception type FileNotFoundException

at ExceptionProcess.Test.main(Test.java:10)
*/

异常处理

异常处理常见手段: try catch finally throws

try catch

  1. 将可能抛出FileNotFoundException 文件不存在异常的代码放在try里

  2. 如果文件存在,就会顺序往下执行,并且不执行catch块中的代码

  3. 如果文件不存在,try 里的代码会立即终止,程序流程会运行到对应的catch块中

  4. e.printStackTrace(); 会打印出方法的调用痕迹,如此例,会打印出异常开始于TestException的第16行,这样就便于定位和分析到底哪里出了异常

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

public class TestException {

public static void main(String[] args) {

File f = new File("d:/LOL.exe");

try{
System.out.println("试图打开 d:/LOL.exe");
new FileInputStream(f);
System.out.println("成功打开");
}
catch(FileNotFoundException e){
System.out.println("d:/LOL.exe不存在");
e.printStackTrace();
}

}
}

使用异常的父类进行catch

FileNotFoundExceptionException的子类,使用Exception也可以catch住 FileNotFoundException

1
2
3
4
5
6
7
8
9
10
11
12
File f= new File("d:/LOL.exe");

try{
System.out.println("试图打开 d:/LOL.exe");
new FileInputStream(f);
System.out.println("成功打开");
}

catch(Exception e){
System.out.println("d:/LOL.exe不存在");
e.printStackTrace();
}

多异常捕捉办法一

有的时候一段代码会抛出多种异常,比如

1
2
new FileInputStream(f);
Date d = sdf.parse("2016-06-03");

这段代码,会抛出 文件不存在异常 FileNotFoundException 和解析异常ParseException
解决办法之一是分别进行catch

1
2
3
4
5
6
7
8
catch (FileNotFoundException e) {
System.out.println("d:/LOL.exe不存在");
e.printStackTrace();

} catch (ParseException e) {
System.out.println("日期格式解析错误");
e.printStackTrace();
}

多异常捕捉办法2

另一个种办法是把多个异常,放在一个catch里统一捕捉

1
2
3
catch (FileNotFoundException | ParseException e) {

}

这种方式从 JDK7开始支持,好处是捕捉的代码更紧凑,不足之处是,一旦发生异常,不能确定到底是哪种异常,需要通过instanceof 进行判断具体的异常类型

1
2
3
4
if (e instanceof FileNotFoundException)
System.out.println("d:/LOL.exe不存在");
if (e instanceof ParseException)
System.out.println("日期格式解析错误");

例如:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class TestException {

public static void main(String[] args) {

File f = new File("d:/LOL.exe");

try {
System.out.println("试图打开 d:/LOL.exe");
new FileInputStream(f);
System.out.println("成功打开");
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
Date d = sdf.parse("2016-06-03");

} catch (FileNotFoundException | ParseException e) {
if (e instanceof FileNotFoundException)
System.out.println("d:/LOL.exe不存在");
if (e instanceof ParseException)
System.out.println("日期格式解析错误");

e.printStackTrace();
}

}
}

finally

无论是否出现异常,finally中的代码都会被执行

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

public class TestException {

public static void main(String[] args) {

File f= new File("d:/LOL.exe");

try{
System.out.println("试图打开 d:/LOL.exe");
new FileInputStream(f);
System.out.println("成功打开");
}
catch(FileNotFoundException e){
System.out.println("d:/LOL.exe不存在");
e.printStackTrace();
}
finally{
System.out.println("无论文件是否存在, 都会执行的代码");
}
}
}

throws

考虑如下情况:

  1. 主方法调用method1
  2. method1调用method2
  3. method2中打开文件
  4. method2中需要进行异常处理
    但是method2不打算处理,而是把这个异常通过throws抛出去
  5. 那么method1就会接到该异常。 处理办法也是两种,要么是try catch处理掉,要么也是抛出去
  6. method1选择本地try catch住 一旦try catch住了,就相当于把这个异常消化掉了,主方法在调用method1的时候,就不需要进行异常处理了
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
package exception;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;

public class TestException {

public static void main(String[] args) {
method1();

}

private static void method1() {
try {
method2();
} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

}

private static void method2() throws FileNotFoundException {

File f = new File("d:/LOL.exe");

System.out.println("试图打开 d:/LOL.exe");
new FileInputStream(f);
System.out.println("成功打开");

}
}

throws与throw区别

throws与throw这两个关键字接近,不过意义不一样,有如下区别:

  1. throws 出现在方法声明上,而throw通常都出现在方法体内

    • throws 表示出现异常的一种可能性,并不一定会发生这些异常;

    • throw则是抛出了异常,执行throw则一定抛出了某个异常对象

异常分类

  • 可查异常(CheckedException)
  • 非可查异常
    • 运行时异常(RunTimeException)
    • 错误(Error)三种分类

可查异常

可查异常即必须进行处理的异常,要么try catch住,要么往外抛,谁调用,谁处理,比如 FileNotFoundException
如果不处理,编译器,就不让你通过

运行时异常

运行时异常RuntimeException指: 不是必须进行try catch的异常

常见运行时异常:

  • 除数不能为0异常:ArithmeticException
  • 下标越界异常:ArrayIndexOutOfBoundsException
  • 空指针异常:NullPointerException

在编写代码的时候,依然可以使用try catch throws进行处理,与可查异常不同之处在于,即便不进行try catch,也不会有编译错误

Java之所以会设计运行时异常的原因之一,是因为下标越界,空指针这些运行时异常太过于普遍,如果都需要进行捕捉,代码的可读性就会变得很糟糕。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
package exception;

public class TestException {

public static void main(String[] args) {

//任何除数不能为0:ArithmeticException
int k = 5/0;

//下标越界异常:ArrayIndexOutOfBoundsException
int j[] = new int[5];
j[10] = 10;

//空指针异常:NullPointerException
String str = null;
str.length();
}
}

错误Error

指的是系统级别的异常,通常是内存用光了

默认设置下,一般java程序启动的时候,最大可以使用16m的内存

如例不停的给StringBuffer追加字符,很快就把内存使用光了。抛出OutOfMemoryError,与运行时异常一样,错误也是不要求强制捕捉的

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
package exception;

public class TestException {

public static void main(String[] args) {

StringBuffer sb =new StringBuffer();

for (int i = 0; i < Integer.MAX_VALUE; i++) {
sb.append('a');
}

}

}

Throwable

Throwable

Throwable是类,Exception和Error都继承了该类
所以在捕捉的时候,也可以使用Throwable进行捕捉
如图: 异常分ErrorException
Exception里又分运行时异常可查异常

Throwable

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
package exception;

import java.io.File;
import java.io.FileInputStream;

public class TestException {

public static void main(String[] args) {

File f = new File("d:/LOL.exe");

try {
new FileInputStream(f);
//使用Throwable进行异常捕捉
} catch (Throwable t) {
// TODO Auto-generated catch block
t.printStackTrace();
}

}
}

自定义异常

创建自定义异常

一个英雄攻击另一个英雄的时候,如果发现另一个英雄已经挂了,就会抛出EnemyHeroIsDeadException
创建一个类EnemyHeroIsDeadException,并继承Exception
提供两个构造方法

  1. 无参的构造方法
  2. 带参的构造方法,并调用父类的对应的构造方法
1
2
3
4
5
6
7
8
9
class EnemyHeroIsDeadException extends Exception{

public EnemyHeroIsDeadException(){

}
public EnemyHeroIsDeadException(String msg){
super(msg);
}
}

抛出自定义异常

在Hero的attack方法中,当发现敌方英雄的血量为0的时候,抛出该异常

  1. 创建一个EnemyHeroIsDeadException实例

  2. 通过throw 抛出该异常

  3. 当前方法通过 throws 抛出该异常

在外部调用attack方法的时候,就需要进行捕捉,并且捕捉的时候,可以通过e.getMessage() 获取当时出错的具体原因

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
package charactor;

public class Hero {
public String name;
protected float hp;

public void attackHero(Hero h) throws EnemyHeroIsDeadException{
if(h.hp == 0){
throw new EnemyHeroIsDeadException(h.name + " 已经挂了,不需要施放技能" );
}
}

public String toString(){
return name;
}

class EnemyHeroIsDeadException extends Exception{

public EnemyHeroIsDeadException(){

}
public EnemyHeroIsDeadException(String msg){
super(msg);
}
}

public static void main(String[] args) {

Hero garen = new Hero();
garen.name = "盖伦";
garen.hp = 616;

Hero teemo = new Hero();
teemo.name = "提莫";
teemo.hp = 0;

try {
garen.attackHero(teemo);

} catch (EnemyHeroIsDeadException e) {
// TODO Auto-generated catch block
System.out.println("异常的具体原因:"+e.getMessage());
e.printStackTrace();
}

}
}

输出:

抛出自定义异常

评论