------- android培训、java培训、期待与您交流! ----------
一.异常的概述
异常:就是程序在运行时出现不正常的情况。
异常由来:问题也是现实生活中一个具体的食物,也可以通过Java类的形式进行描述,并封装成对象。其实就是java对不正常情况进行描述后的对象体系。
二.异常体系
Throwable
|--Error
|--Exception
有两种对问题的划分方式:
一种是严重的问题;
一种是非严重的问题。
对于Error一般不编写针对性的代码对其进行处理。
对于非严重的,java通过Exception类进行描述。
对于Exception可以使用针对性的处理方式进行处理。(此处重点也是围绕Exception和其子类讲述其处理方式)
三、异常的处理
1、 java提供了特有的语句进行处理。
try
{
需要被检测的代码。
}
catch(异常类 变量)
{
处理异常的代码;(处理方式)
}
finally
{
一定会执行的语句;
}
有三个结合格式:
a、try
{
}
catch ()
{
}
b、try
{
}
finally
{
}
c、try
{
}
catch ()
{
}
finally
{
}
package exceptionpackage;
class Demo
{
int div(int a,int b)
{
return a/b;
}
}
public class ExceptionDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
Demo d=new Demo();
try
{
int x=d.div(4, 0);//检测到异常
System.out.println("x="+x);//此句不再执行
}
catch(Exception e) //抓到异常
{
System.out.println("除零啦");
}
System.out.println("over");//处理异常后继续往下执行
}
}
输出结果:
2.Throwable中的方法:
1.getMessage():获取异常信息,返回字符串。
2.toString():获取异常类名和异常信息,返回字符串。
3.printStackTrace():获取异常类名和异常信息,以及异常出现在程序中的位置,返回值void。
4.printStackTrace(PrintStreams):通常用该方法将异常内容保存在日志文件中,以便查阅。
3.异常声明:当功能创建者通过throws在功能上声明了异常,此功能的调用者必须要处理,可try可抛。
声明异常时,建议声明更为具体的异常,这样处理的可以更具体。
class Demo
{
int div(int a,int b) throws Exception //声明异常,表示该功能有可能会出现问题。
{
return a/b;
}
}
4.多异常的处理
<span style="font-size:18px;">package exceptionpackage;
class Demo
{
int div(int a,int b) throws ArithmeticException,ArrayIndexOutOfBoundsException//声明可能出问题的两个异常
{
int arr[] =new int[a];
System.out.println(arr[4]);
return a/b;
}
}
public class ExceptionDemo {
public static void main(String[] args)
{
// TODO Auto-generated method stub
Demo d=new Demo();
int x;
try {
x = d.div(6, 0);
System.out.println("x="+x);
}
catch(ArrayIndexOutOfBoundsException e)//捕捉数组角标越界异常
{
e.printStackTrace();
System.out.println("角标越界了");
}
catch (ArithmeticException e) { //捕捉算术异常
// TODO Auto-generated catch block
System.out.println(e.toString());
System.out.println("被零除了");
}
System.out.println("over");
}
}
</span>
注意:声明异常时,建议声明更为具体的异常,这样处理的可以更具体
对方声明几个异常,就对应有几个catch块,不要定义多余的catch块。
如果有多个catch块中异常出现继承关系,父类异常catch块放在最下面。
建议在进行catch处理时,catch中一定要定义具体处理方式。
不要简单定义一句 e.printStackTrace()
也不要简单的书写一条输出语句。
四.自定义异常
因为项目中会出现特有的问题,而这些问题并未被java所描述并封装对象。所以对于这些特有的问题可以按照java对问题封装的思想将特有的问题,进行自定义的异常封装。
需求:在本程序中,对于除数是负数,也视为是错误的是无法进行运算的,那么久需要对这个问题进行自定义描述。
<pre name="code" class="java">package exceptionpackage;
class FushuException extends Exception
{
FushuException(String msg)
{
super(msg);
}
}
class Demo1
{
int div(int a,int b) throws FushuException //函数内部出现异常,一般要声明
{
if(b<0)
throw new FushuException("出现了除数的负数的情况/by fushu");//手动通过throws关键字抛出自定义异常。
return a/b;
}
}
public class ZidingyiException {
public static void main(String[] args) {
Demo1 d=new Demo1();
try {
int x= d.div(3, -8);
System.out.println(x);
} catch (FushuException e) {
// TODO Auto-generated catch block
System.out.println(e.toString());
}
System.out.println("over");
}
}
结果:
当在函数内部出现了throw抛出异常对象,那么就必须要给对应的处理动作要么在内部 try catch处理,
要么在函数上声明让调用着处理。
一般情况下,函数内部出现异常,函数上需要声明。
如何定义异常信息呢?
因为弗雷中已经把异常信息的操作都完成了,所以子类只要在构造时,将异常信息
传递给父类通过super语句,那么就可以直接通过getMessage方法获取自定义的异常。
注意:自定义异常必须是自定义类继承Exception.
继承Exception的原因:
异常体系有一个特点,因为异常类和异常对象都需要被抛出,他们都具备可抛性,这个
可抛性是throwable这个体系中的独有特点,只有这个体系中的类和对象才可以被throws和
throw操作。
五. throws 和throw的区别
throws使用在函数上,throw使用在函数内。
throws后面跟的是异常类,可以跟多个,用逗号隔开,throw后面跟的是异常对象。
六.RuntimeException
Exception中有一个特殊的子类异常RuntimeException运行时异常。如果在函数内抛出该异常,函数上可以
不用声明,编译一样通过。如果在函数上声明了该异常,调用者可以不用进行处理,编译一样通过。
package exceptionpackage;
class Demo3
{
int div(int a,int b)
{
if(b==0)
throw new ArithmeticException("除数不能为零");
return a/b;
}
}
public class RuntimeException {
public static void main(String[] args) {
Demo3 d=new Demo3();
System.out.println(d.div(3, 0));
System.out.println("over");
}
}
结果:
对于异常分为两类:
1,编译是被检测的异常。
2,编译是不被检测的异常(运行时异常,RuntimeException以其子类)
一个列子:
<span style="font-size:18px;">/*
* 毕老师用电脑上课
* 上课会出现的问题:电脑冒烟,电脑蓝屏
* 对问题进行描述,封装成对象
*
* 可是当冒烟发生后,出现讲课进度无法继续,这个时候出现了讲师的问题:课时计划无法完成
*
*/
//电脑蓝屏异常
class LanpingExcepion extends Exception
{
LanpingExcepion(String message)
{
super(message);
}
}
//电脑冒烟异常
class MaoyanException extends Exception
{
MaoyanException(String message)
{
super(message);
}
}
//由于冒烟导致的课时无法完成异常
class NoplanException extends Exception
{
public NoplanException(String message) {
// TODO Auto-generated constructor stub
super(message);
}
}
class Computer
{
private int state=3;
public void run() throws LanpingExcepion, MaoyanException
{
if(state==2)
{
throw new LanpingExcepion("电脑蓝屏了");
}
if(state==3)
throw new MaoyanException("电脑冒烟了");
System.out.println("电脑运行");
}
void reset()
{
state=1;
System.out.println("电脑重启");
}
}
class Teacher
{
private String name;
private Computer comp;
Teacher(String name)
{
this.name=name;
comp=new Computer();
}
void prelect() throws NoplanException //未处理的异常要声明
{
try
{
comp.run();
}catch(LanpingExcepion e)
{
comp.reset();
}
catch(MaoyanException e)
{
test();
throw new NoplanException("课时无法继续"+e.getMessage());
}
System.out.println("讲课");
}
public void test()
{
System.out.println("做练习");
}
}
public class Example {
public static void main(String[] args) {
Teacher t=new Teacher("毕老师");
try {
t.prelect();
} catch (NoplanException e) {
// TODO Auto-generated catch block
System.out.println(e.toString());
System.out.println("换老师,或者放假");
}
}
}</span>
</span>
finally代码块:定义一定执行的代码,通常用于关闭资源。
记住一点:catch是用于处理异常,如果没有catch就代表异常没有被处理过,如果该异常是检测时异常,需要声明出去。
六.异常在父类覆盖中的体现:
1,子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常或者该异常的子类。
2,如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子类。
3,如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方式时也不可以抛。
如果子类方法发生了异常,就必须要进行try处理,绝对不能抛。
练习:
/*
* 练习:有一个圆形和长方形,都可以获取面积,对于面积如果出现非法的数值,视为出现获取面积出现问题,问题通过异常来表示。
* 现有对这个程序进行基本设计。
*/
package exceptionpackage;
//自定义异常继承RuntimeException
class NoValueException extends RuntimeException
{
public NoValueException(String message) {
// TODO Auto-generated constructor stub
super(message);
}
}
//图形接口
interface Shape
{
void getArea();
}
class Rec implements Shape
{
private int len,wid;
Rec(int len,int wid)
{
if(len<0|| wid<0)
throw new NoValueException("出现非法值");//抛出RuntimeException子类异常不需要声明
this.len=len;
this.wid=wid;
}
public void getArea()
{
System.out.println(len*wid);
}
}
class Circle implements Shape
{
public static final double PI=3.14;
private int radius;
Circle(int radius)
{
if(radius<=0)
throw new NoValueException("非法值");
this.radius=radius;
}
public void getArea() {
System.out.println(radius*radius*PI);
}
}
public class ExceptionTest {
public static void main(String[] args) {
// TODO Auto-generated method stub
Rec r=new Rec(3,5);
r.getArea();
Circle c=new Circle(-3);
c.getArea();
}
}
结果: