什么是异常?让我们先运行一段代码来理解
class Demo
{
int chuFa(int x,int y)//这是一个除法运算
{
return x/y;
}
}
public class ExceptionDemo {
public static void main(String[] args)
{
Demo d = new Demo();
int a = d.chuFa(4,1);//这里传入的除数是1
System.out.println(a);
System.out.println("over");
}
}
运行结果如下:
而当我把主函数中的除数换成0的话,我们知道在数学中这是没有意义的运算。在Java运行时也会运行失败,提示如下:
结果表明,在程序运行的过程中出现了不正常的情况,
是什么情况不正常呢?提示中给出,ArithmeticException:/ by zero 算术异常:被0除了
我们把这样在程序中出现的不正常的情况称之为异常。
异常:就是程序在运行时出现的不正常情况。
异常由来:问题现实生活中一个具体的事物,也可以通过Java类的形式进行描述,并封装成对象。其实就是Java对不正常情况进行描述后的对象体现。(简单说就是把问题封装成了对象)(这里又体现了Java语言的封装性和面向对象的特征)
对于问题的划分,有两种,一种是严重的问题,一种是非严重的问题
对应于严重的,Java通过Error类进行描述
对于非严重的,Java通过Exception类进行描述
对于Exception,一般都会编写针对性的代码进行处理
无论Error还是Exception都具有一些共性的内容。
比如:不正常情况的信息,引发原因等。
所以,Error和Exception都有一个父类叫做Throwable
异常最常见的处理方法如下:
try
{
需要被检测的代码
}
catch (异常类 变量)
{
异常的处理方式
}
finally
{
一定会执行的语句
}
try catch方法处理异常的流程
当出现异常情况后,会创建一个异常对象,这个异常情况被try检测到以后就会抛给catch,catch会做出对应的解决方式,处理完异常以后,之后的代码还可以运行到。把之前的代码加上异常处理机制后就是:
public class ExceptionDemo {
public static void main(String[] args)
{
Demo d = new Demo();
try
{
int a = d.chuFa(4,0);//这里传入的除数是1
System.out.println(a);
}
catch (Exception e)
{
System.out.println("除零啦");
}
System.out.println("over");
}
}
运行结果就变成了:
从结果中可以看出,异常处理完之后,会继续执行下边的语句
对于异常常见的几种获取信息的方法:
String getMessage();//引发错误的原因
String toString();//异常类+引发错误的原因
void printStackTrace();//异常类+引发错误的原因+追踪错误的位置信息
声明异常用throws关键字
由于在调用方法的时候会有异常发生,导致程序停止,所以在定义方法的时候要用throws关键字声明一下会有异常情况,意思就是告诉调用该方法的人,这个方法可能会发生异常,你需要处理
作用:提高安全性,让调用和作出处理,不处理则编译失败。
对于调用了已声明了异常的方法处理方式
一旦调用了已经声明异常的方法,就必须对其作出处理,否则会编译失败
处理方式有两种:
throws :抛出,由抛出函数的外层函数进行处理,如果处理不了,再次声明抛出可以继续往外抛,那么,抛到哪里是个头呢?
jvm虚拟机,虚拟机接收到抛出的异常之后,会使用默认的方法,让程序停掉,
try:
多异常的处理方式
class Demo
{
int chuFa(int x,int y) throws ArithmeticException,ArrayIndexOutOfBoundsException //在这个方法里声明了两个异常
{
int[] arr = new int[x];
System.out.println(arr[4]);//如果数组角标越界的话会触发角标越界异常
return x/y;
}
}
public class ExceptionDemo2 {
public static void main(String[] args)
{
Demo d = new Demo();
try{
int a = d.chuFa(5,1);//这里传入参数容易引发异常
System.out.println(a);
}
catch(ArithmeticException e)//触发ArithmeticException异常的时候执行的代码块
{
System.out.println("除零了");
}
catch(ArrayIndexOutOfBoundsException e)//触发ArrayIndexOutOfBoundsException异常的时候执行的代码块
{
System.out.println("角标越界啦");
}
catch(Exception e)//这是父类异常的代码块
{
System.out.println("hahaha:"+e.toString());
}
System.out.println("over");
}
}
这个时候没有发生任何异常,运行会正常结束:
当把chuFa方法中的参数改为4,1时(创建一个角标越界的异常)结果如下:
当把参数改为5,0时,(创建一个运算异常) 结果如下:
我们可以看到,放在最下边的catch块,虽然是每个异常的父类,但并没有执行,
因为代码是从上到下执行的,找到了符合条件的catch块,就不会去找其他的catch块,这样写父类的异常catch块是为了避免其他异常蹦出来中断程序,可以编译通过,也可以运行但是不建议这么写,因为在运行过程中出了啥异常都会被父类异常处理掉,如果出现了别的异常,程序员需要知道;
建议:在进行异常处理时,catch中一定要有具体的定义方式,不要简单的定义一句打印异常信息,也不要简单的书写一条输出语句
自定义异常
在后期做开发的时候,描述具体的事物有具体的要求,也就是说会遇到不同的异常,而这些是Java没有进行过描述并封装对象的
所以,对于这些特有的问题,我们可以按照Java的封装思想,将特有的问题进行自定义的异常封装。
怎么样进行异常的封装呢?我们用代码来体现:
//需求:定义一个除法运算,除了不能除零外,还要定义不能除以负数
class FuShuException extends Exception//定义一个异常类(对象),需要继承Exception
{
private int value;//想把出错误的数字也获取到,在这里创建一个value
FuShuException(String msg,int b)//因为自定义异常还没有定义自己的异常信息,所以在初始化的时候就要传一个描述性的文字,作为他的异常信息描述语句
{
super(msg);
//Exception类中已经定义了自定义异常信息的方法,这里把参数传给父类Exception就行了
//下边调用toString方法的时候会自动调用getMessage方法,将描述语句打印出来
this.value = b;
}
public int getValue()
{
return value;
}
}
class Demo
{
int chuFa(int x,int y) throws FuShuException,ArithmeticException
{
if (y<0)//除数小于0,创建异常对象并抛出
{
throw new FuShuException("除数为负数了 / by fushu",y);
}
return x/y;
}
}
public class ExceptionDemo3 {
public static void main(String[] args)
{
Demo d = new Demo();
try//对自定义的异常进行处理
{
int a = d.chuFa(4,-1);
System.out.println(a);
}
catch (FuShuException e)
{
System.out.println(e.toString()+"---错误的数字是"+e.getValue());
}
catch(ArithmeticException e)
{
System.out.println(e.toString());
}
}
}
运行结果为:
我们可以看到,自定义异常,必须是自定义类继承Exception
继承Exception 的原因:
异常体系中有一个特点,因为异常类和对象都被抛出
他们都具备可抛性。这个可抛性是Throwable这个体系中的独有特点
注意的是,在自定义异常的时候,没有必要所有的信息都自己去做,
比如异常信息的操作,在父类中就已经做完了,父类中有这样的方法:
而在Exception类中,也有相同的方法:
所以我们这里只要把自定义的信息用super语句传给父类。
就是以上代码中的 “super(msg);”
就可以直接通过getMessage方法获取自定义的异常信息。
throw和throws的区别
throws使用在函数上,throw使用在函数内
throws后边跟的是异常类,可以跟多个,用逗号隔开
throw后跟的是异常对象
运行时异常
Exception中有一个特殊的子类异常RuntimeException
如果在函数中抛出该异常,函数上可以不用声明,编译一样通过。
如果在函数上声明了该异常,调用者可以不用进行处理,编译一样通过。
之所以不用在函数声明,是因为不需要让调用者处理
当该异常发生,希望程序停止,因为在运行时,出现了无法继续运算的情况,希望程序停止后,对代码进行修正。
子类例如:数组角标越界异常(角标越界都找不到元素,没有办法继续运算下去了),空指针异常(对象没有指向的示例化对象,也无法进行下一步的运算)
这些情况下,我们希望程序停下来,不希望catch处理之后继续再进行运算,
因为再运算也没有什么意义了。
用一段小代码来具体体现一下RunTimeException的特点:
//运行时异常:
//需求:定义一个除法运算,除了不能除零外,还要定义不能除以负数
//一旦发生除零,除负数的情况,应该让程序停下来,因为继续算也没什么 意义了
class FuShuException extends RuntimeException//该种异常我们不会处理,发生该异常的时候希望程序停下来,所以这里要继承RunTimeException
{
FuShuException(String msg)
{
super(msg);
}
}
class Demo
{
int chuFa(int x,int y)//因为里边抛出的是RunTimeException,所以在这里可以不用声明,在主函数中也不用处理
{
if (y<0)
{
throw new FuShuException("除数为负数了 / by fushu");
}
if (y==0)
{
throw new ArithmeticException("除数为0,没意义");
}
return x/y;
}
}
public class ExceptionDemo4 {
public static void main(String[] args)
{
Demo d = new Demo();
int a = d.chuFa(4,0);
System.out.println(a);
}
}
在自定义异常的时候:如果该异常的发生无法再继续进行运算,就让自定义异常继承RunTimeException。
对于异常分为两种:
1,编译时被检测的异常。
2,编译时不被检测的异常(运行时异常,RunTimeException以及其子类)
所以在分析问题定义异常的时候,先要分析该异常能不能被处理,
|—–如果不能处理,需要修正代码的时候,就继承RunTimeException,
|—–如果该异常可以被try catch处理,就继承Exception。
一个小练习:老师带电脑上课的过程中异常处理:
//异常练习
/*
需求:毕老师讲课,要用电脑讲课,
而会时常有异常,比如:电脑蓝屏了,:处理方式,重启,继续上课
电脑冒烟了。:处理方式,又有了一个新的异常:
课时无法完成,处理方式:换老师或者放假
思想:封装,将各种异常封装成对象,关键字提取;
*/
class LanPingException extends Exception//蓝屏异常
{
LanPingException(String msg)
{
super(msg);
}
}
class MaoYanException extends Exception//冒烟异常
{
MaoYanException(String msg)
{
super(msg);
}
}
class NoPlanException extends Exception//课时无法继续异常
{
NoPlanException(String msg)
{
super(msg);
}
}
class Teacher
{
private String name;
Teacher(String name)
{
this.name = name;
}
public void jiangKe() throws NoPlanException//老师讲课过程中会抛出课时无法继续异常,所以在这里标识一下
{
//老师来讲课,带上了自己的电脑,开启电脑之后开始讲课
Computer cmpt = new Computer();
try
{
cmpt.run();//而电脑会出现异常,所以在这里调用run方法的时候要对异常进行处理
}
catch (LanPingException e)
{
cmpt.restart();
}
catch (MaoYanException e)
{
//发生冒烟异常的时候,处理方法应该是先让学生们做练习,然后给调用讲课方法的人抛一个课时无法完成异常
//交给上一层领导处理
test();
throw new NoPlanException("课时无法继续");
}
System.out.println("开始讲课");
}
public void test()
{
System.out.println("做练习");
}
}
class Computer
{
private int state = 3;
public void run() throws MaoYanException,LanPingException
{
//电脑的诸多自定义异常都要在这里产生异常对象
if(state==2)
{
throw new LanPingException("蓝屏了");
}
if (state==3)
{
throw new MaoYanException("冒烟了");
}
System.out.println("电脑运行");
}
public void restart()
{
System.out.println("电脑重启");
}
}
public class ExceptionTest{
public static void main(String[] args)
{
Teacher t = new Teacher("毕老师");
try
{
t.jiangKe();//在调用讲课方法的时候,接收到课时无法完成异常,将在这里进行处理
}
catch (NoPlanException e)
{
System.out.println("换老师或者放假-----"+e.getMessage());
}
}
}
运行结果如下:
当代码中state为1的时候(电脑没有发生任何异常的时候)结果为:
当代码中state为2的时候(电脑发生蓝屏异常的时候)结果为:
当代码中state为3的时候(电脑发生冒烟异常的时候)结果为:
异常——finally
finally代码块,定义一定会执行的代码,通常用于关闭资源
在数据库操作中最为常见的例子:
public void method()
{
连接数据库;
数据操作//throw new SQLException如果抛出异常,下面的代码就不会运行了,也就是说资源没有被关闭,
关闭数据库//而这一步是必须要执行的
}
该代码中如果异常出现,则不会关闭资源,这不是我们想要的。
解决这样的问题可以这么做:
public void method() throws 没有操作成功异常
{
try
{
连接数据库;
数据操作;//throw new 数据操作异常;
}
catch(数据操作异常 e)
{
会对数据库异常进行处理;
throw new 没有操作成功异常;
}
finally
{
关闭数据库;
}
}
在Java中,数据存储操作是调用数据库的方法完成的,
我要存数数据,把数据给你即可,至于你是怎么存的,我没必要知道,
但是我必须知道的是:数据有没有存储成功,
就像在以上的代码中,数据库操作出现异常,应该在数据库内部直接处理,
而不是抛出来异常,因为抛出来我也看不懂,所以在你内部直接处理掉就可以了。
但是你要告诉我,有没有存储成功,也就是说这里可以抛出来一个“没有操作成功异常”这个我可以处理。
而不管有没有存储成功,资源是必须要关闭的,所以我们在这里把关闭资源的动作放在finally语句中。
异常处理语句的其他格式:
//第一种格式:
try
{
}
catch()
{
}
//第二种格式:
try
{
}
caatch()
{
}
finally
{
}
//第三种格式
try
{
}
finally
{
}
记住一点:catch是用来处理异常的,如果没有catch代表该异常没有被处理过,如果该异常时检测时异常,那么必须申明。
异常在子父类覆盖中的体现
//异常之间的关系是:
// |--Exception
// |--AException
// |--BException
// |--CException
class AException extends Exception
{
}
class BException extends AException
{
}
class CException extends Exception
{
}
class Fu
{
public void show() throws AException//父类方法抛出一个AException
{
}
}
class Zi extends Fu//子类继承父类
{
public void show()throws BException//在这里抛出一个AException(也可以抛出BException,但是不能抛出CException)
//如果这里真的会冒出一个C异常的话,只能try,绝对不能抛
{
}
}
class Test1
{
public void function(Fu f)
{
try
{
f.show();
}
catch (AException e)
{
}
}
}
public class Test {
public static void main(String[] args)
{
Test1 t1 = new Test1();
t1.function(new Zi());
}
}
通过一串小代码来更加熟练的运用一下异常的处理机制:
//长方形圆形的面积计算程序
/*
首先思考:长方形,圆形都是图形,都有获取面积的这样一个功能,
但是长和宽是长方形特有的,半径是圆特有的属性
可以将获取面积的方法封装在一个接口中
*/
interface Shape
{
void getArea();
}
//长方形的计算过程:
class ChangFangXing implements Shape
{
private int chang;
private int kuang;
ChangFangXing(int chang,int kuang)
{
this.chang = chang;
this.kuang = kuang;
}
public void getArea()
{
//如果给出的值小于0的话,必须停掉程序,不可以再往下运行了。
if (chang<=0 || kuang<=0)
{
throw new NoValueException("长或宽的值小于0!!!");
}
System.out.println(chang*kuang);
}
}
//圆的面积计算过程:
class Yuan implements Shape
{
private int radiu;
//PI的值是不变的,所以在这里把他定义成常量
public static final double PI = 3.14;
Yuan(int radiu)
{
this.radiu = radiu;
}
public void getArea()
{
//同样的,如果他的半径小于0,程序停掉
if (radiu<=0)
{
throw new NoRadiuException("圆的半径值出错!!!");
}
System.out.println(radiu*radiu*PI);
}
}
class NoValueException extends RuntimeException
{
NoValueException(String msg)
{
super(msg);
}
}
class NoRadiuException extends RuntimeException
{
NoRadiuException(String msg)
{
super(msg);
}
}
public class ExceptionTest2 {
public static void main(String[] args)
{
//创建一个长方形对象,求面积
ChangFangXing cfx = new ChangFangXing(3,4);
cfx.getArea();
//创建一个圆的对象,并求面积
Yuan y = new Yuan(2);
y.getArea();
}
}
class MyException extends Exception
{
MyException(String message)
{
super(message);
}
}
try
{
throw new AException();
}
catch(AException e)
{
throw e
}
如果该异常处理不了,但并不属于该功能出现的异常。
可以将异常转换后,再抛出和该功能相关的异常。
或者异常可以处理,但需要将异常产生的和本功能相关的问题提供出来
让调用者知道,并处理,也可以将捕获异常处理后,转换新的异常抛出
try
{
throw new AException();
}
catch(AException e )
{
//对AException处理。
throw new BException();
}
比如:汇款的例子。
我去给张三汇款,但是出了异常没有汇成功,汇款机需要先将我汇的钱再存进我的卡里,然后告诉我,没有汇成功,我再去处理这个没有汇成功的异常(换个银行或者换台机子)