文章目录
异常的定义
异常定义:导致程序的正常流程被中断的事件,叫做异常
文件不存在异常:
比如要打开d盘的LOL.exe
文件,这个文件是有可能不存在的。Java中通过 new FileInputStream(f)
试图打开某文件,就有可能抛出文件不存在异常FileNotFoundException
如果不处理该异常,就会有编译错误。处理办法参见 异常处理
ackage 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");
//试图打开文件LOL.exe,会抛出FileNotFoundException,如果不处理该异常,就会有编译错误
new FileInputStream(f);
}
}
异常处理
异常处理的常见方法:
try catch finally throws
try catch
1.将可能抛出FileNotFoundException
文件不存在异常的代码放在try
里
2.如果文件存在,就会顺序往下执行,并且不执行catch
块中的代码
3. 如果文件不存在,try
里的代码会立即终止,程序流程会运行到对应的catch
块中
4. e.printStackTrace();
会打印出方法的调用痕迹,如此例,会打印出异常开始于TestException
的第16行,这样就便于定位和分析到底哪里出了异常
package j2se;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
public class lian {
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("文件不存在");
e.printStackTrace();
}
}
}
使用异常的父类进行catch
FileNotFoundException
是Exception
的子类,使用Exception
也可以catch
住FileNotFoundException
catch(Exception e){
System.out.println("d:/LOL.exe不存在");
e.printStackTrace();
多异常捕捉办法
有时,一段代码会泡出多种异常,如:
new FileInputStream(f);
Date d = sdf.pase("2016-06-03");
代码会抛出文件不存在异常FileNotFoundException
和解析异常ParseException
.
第一种解决办法是:
对抛出的异常分别进行catch
catch (FileNotFoundException e) {
System.out.println("d:/LOL.exe不存在");
e.printStackTrace();
} catch (ParseException e) {
System.out.println("日期格式解析错误");
e.printStackTrace();
}
第二种解决办法是:
把多个异常,放在一个catch里统一捕捉
catch (FileNotFoundException | ParseException e) {}
这种方式从 JDK7开始支持,好处是捕捉的代码更紧凑,不足之处是,一旦发生异常,不能确定到底是哪种异常,需要通过instanceof
进行判断具体的异常类型
if (e instanceof FileNotFoundException)
System.out.println("d:/LOL.exe不存在");
if (e instanceof ParseException)
System.out.println("日期格式解析错误");
finally
无论是否出现异常,finally
中的代码都会被执行,具体来说:
1、如果没有异常发生,在try内的代码执行结束后执行。
2、如果有异常且被catch捕获,在catch内的代码执行结束后执行。
3、如果有异常但是没有发生捕捉,则在异常被抛出给上层之前执行。
由于finally这个特点,它一般用于释放资源,如数据库连接,文件流等。、
try/catch/finally语法中,catch是不必须的,也就是可以只有try/finally,表示不捕获异常,异常自动向上传递,但是finally中的代码在异常发生后,抛出给上层前执行。
try、catch、finally中有return的情况:
1、try或者里面有return语句,则return语句在finally语句执行结束后才执行,但是finally并不能改变返回值。
2、finally 中如果也有return,try和catch内的return会丢失,实际返回finally中的返回值。finally中的return不仅会覆盖try和catch内的return值,还会掩盖try和catch内的异常,就像没有发生异常一样。
throws
考虑如下情况:
主方法调用method1
–>method1
调用method2
–>method2
中打开文件
method2
中需要进行异常处理
但是method2
不打算处理,而是把这个异常通过throws
抛出去。那么method1
就会接到该异常。 处理办法也是两种,要么是try catch
处理掉,要么也是抛出去。
method1
选择本地try catch
住 ,一旦try catch
住了,就相当于把这个异常消化掉了,主方法在调用method1
的时候,就不需要进行异常处理了
package j2se;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
public class lian {
public static void main(String[] args) {
mothod1();
}
private static void mothod1(){
try{
mothod2();
}catch (FileNotFoundException e){
e.printStackTrace();
}
}
private static void mothod2() throws FileNotFoundException{
File f = new File("d:LOL,exe");
System.out.println("试图打开文件");
new FileInputStream(f);
System.out.println("打开成功");
}
}
throw和throws的区别
throws与throw这两个关键字接近,不过意义不一样,有如下区别:
- throws 出现在方法声明上,而throw通常都出现在方法体内。
- throws 表示出现异常的一种可能性,并不一定会发生这些异常;throw则是抛出了异常,执行throw则一定抛出了某个异常对象。
{.....
throw new EnemyHeroIsDeadException(h.name + " 已经挂了,不需要施放技能" );}
异常分类
异常分类: 可查异常,运行时异常和错误3种
其中,运行时异常和错误又叫非可查异常
可查异常
可查异常: CheckedException
可查异常即必须进行处理的异常,要么try catch
住,要么往外抛,谁调用,谁处理,比如 FileNotFoundException
如果不处理,编译器,就不让你通过
运行时异常
运行时异常
RuntimeException
指: 不是必须进行try catch
的异常 常见运行时异常:
除数不能为0异常:ArithmeticException
下标越界异常:ArrayIndexOutOfBoundsException
空指针异常:NullPointerException
在编写代码的时候,依然可以使用try catch throws
进行处理,与可查异常不同之处在于,即便不进行try catch
,也不会有编译错误
Java之所以会设计运行时异常的原因之一,是因为下标越界,空指针这些运行时异常太过于普遍,如果都需要进行捕捉,代码的可读性就会变得很糟糕。
//任何除数不能为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
与运行时异常一样,错误也是不要求强制捕捉的
StringBuffer sb =new StringBuffer();
for (int i = 0; i < Integer.MAX_VALUE; i++) {
sb.append('a');
}
总体上异常分三类:
- 错误
- 运行时异常
- 可查异常
问题:
- 运行时异常
RuntimeException
,能否被捕捉?
错误Error
,能否被捕捉? - 面试题常问题: 运行时异常与非运行时异常的区别
1.运行时异常和错误都可以被捕捉。
(1)运行时异常
都是RuntimeException
类及其子类异常,如NullPointerException
、IndexOutOfBoundsException
等,这些异常是不可查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。
当出现RuntimeException
的时候,我们可以不处理。当出现这样的异常时,总是由虚拟机接管。比如:我们从来没有人去处理过NullPointerException
异常,它就是运行时异常,并且这种异常还是最常见的异常之一。
出现运行时异常后,如果没有捕获处理这个异常(即没有catch
),系统会把异常一直往上层抛,一直到最上层,如果是多线程就由Thread.run()
抛出,如果是单线程就被main()
抛出。抛出之后,如果是线程,这个线程也就退出了。如果是主程序抛出的异常,那么这整个程序也就退出了。运行时异常是Exception
的子类,也有一般异常的特点,是可以被catch
块处理的。只不过往往我们不对他处理罢了。也就是说,你如果不对运行时异常进行处理,那么出现运行时异常之后,要么是线程中止,要么是主程序终止。
如果不想终止,则必须捕获所有的运行时异常,决不让这个处理线程退出。队列里面出现异常数据了,正常的处理应该是把异常数据舍弃,然后记录日志。不应该由于异常数据而影响下面对正常数据的处理。
(2)非运行时异常(是可查异常)
是RuntimeException
以外的异常,类型上都属于Exception
类及其子类。如IOException
、SQLException
等以及用户自定义的Exception
异常。对于这种异常,JAVA编译器强制要求我们必需对出现的这些异常进行catch
并处理,否则程序就不能编译通过。所以,面对这种异常不管我们是否愿意,只能自己去写一大堆catch
块去处理可能的异常。
Throwable
Throwable
是类,Exception
和Error
都继承了该类
所以在捕捉的时候,也可以使用Throwable
进行捕捉
如图: 异常分Error
和Exception
Exception里又分运行时异常和可查异常。
File f = new File("d:/LOL.exe");
try {
new FileInputStream(f);
//使用Throwable进行异常捕捉
} catch (Throwable t) {
// TODO Auto-generated catch block
t.printStackTrace();
}
问题:
在方法声明上,可以抛出指定的异常,比如FileNotFoundException
那么能否抛出Throwable
这个类?
这个方法的调用者又该如何处理?
可以抛出Throwable这个类,但catch的时候必须也是Throwable类型
package exception;
import java.io.FileNotFoundException;
public class demo1 {
public static void main(String[] args){
try {
throwCheck(2);
}catch (Throwable e){
e.printStackTrace();
}
throwRun(-3);
}
public static void throwCheck(int a) throws Throwable{
if(a > 0){
throw new Exception("a");
}
}
public static void throwRun(int a){
if(a > -1){
throw new RuntimeException("a > -1");
}
}
}
自定义异常
创建自定义异常
一个英雄攻击另一个英雄的时候,如果发现另一个英雄已经挂了,就会抛出EnemyHeroIsDeadException
创建一个类EnemyHeroIsDeadException,并继承Exception
提供两个构造方法
- 无参的构造方法
- 带参的构造方法,并调用父类的对应的构造方法
class EnemyHeroIsDeadException extends Exception{
public EnemyHeroIsDeadException(){
}
public EnemyHeroIsDeadException(String msg){
super(msg);
}
}
抛出自定义异常
在Hero的attack
方法中,当发现敌方英雄的血量为0的时候,抛出该异常
- 创建一个
EnemyHeroIsDeadException
实例 - 通过
throw
抛出该异常 - 当前方法通过
throws
抛出该异常
在外部调用attack方法的时候,就需要进行捕捉,并且捕捉的时候,可以通过e.getMessage() 获取当时出错的具体原因
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();
}
}
}
练习-异常综合1 :
这是一个类图
Account类: 银行账号
属性: balance 余额
方法: getBalance() 获取余额
方法: deposit() 存钱
方法: withdraw() 取钱
OverdraftException: 透支异常,继承Exception
属性: deficit 透支额
package yinhan;
public class Account {
protected double balance;
protected double deficit;
public Account(double balance,double deficit) {
this.balance = balance;
this.deficit = deficit;
}
class OverdraftException extends Exception{
public OverdraftException(){}
public OverdraftException(String msg){
super(msg);
}
}
public double getBalance(){
return this.balance;
}
public void deposit(double money){
balance = balance + money;
}
public void withdraw(double draftMoney) throws OverdraftException {
if (balance <=0 || balance+deficit < draftMoney){
throw new OverdraftException("您的账户余额不足,剩余余额为:"+this.balance);
}
else{
balance = balance - draftMoney;
System.out.println("账户余额为:"+this.balance);
}
}
public static void main(String[] args){
Account A = new Account(1000,1000);
System.out.println(A.getBalance());
try{
A.withdraw(2100);
}catch(OverdraftException e){
e.printStackTrace(); }
}
}