异常
class Demo
{
public int div(int a,int b)
{
return a/b;
}
}
class ExceptionDemo
{
public static void main(String[] args)
{
Demo d=new Demo();
// d.div(3,0);
byte[] bt=new byte[1024*1024*800];
}
}
1、异常:就是程序在运行时出现不正常情况。
异常由来:问题也是现实生活中一个具体的事物,也可以通过Java的类的形式进行描述,并封装成对象,其实就是Java对不正常情况进行描述后的对象体现。
对于问题的划分:两种:一种是严重的问题,一种是非严重问题。
对于严重问题,Java通过Error进行描述。
对于Error一般不编写针对性的代码对其进行处理。
对于非严重问题,Java通过Exception进行描述。
对于Exception可以使用针对性的处理方式进行处理。
无论Error或者Exception都具有一些共性内容
比如:不正常的信息,引发原因等。
throwable
---Error
---Exception
--RuntimeException
2、异常的处理
Java提供了特有的语句进行处理。
try
{
需要检测的代码
}catch(异常类 变量)
{
处理异常的代码(处理方式)
}
finally
{
一定会执行的语句。
}
3、对捕获到的异常对象进行常见方法操作
String getMessage();
class Demo
{
public int div(int a,int b) throws Exception //在功能上通过throws的关键字声明了该功能有可能会出现问 //题,
{
return a/b;
}
}
class ExceptionDemo
{
public static void main(String[] args) // throws Exception
{
Demo d=new Demo();
try
{
int x=d.div(3,0);
System.out.println("x="+x);
}
catch (Exception e) //Exception e=new ArithmeticException();
{
System.out.println("BY ZErO");
System.out.println(e.getMessage()); //by zero
System.out.println(e.toString()); //异常名称:异常信息
e.printStackTrace(); //异常名称:异常信息,异常出现的位置
//其实jvm默认的异常处理机制,就是在 //调用printStackTrace()方法,打印异常的堆栈的跟踪信息
}
System.out.println("over");
//byte[] bt=new byte[1024*1024*800];
}
}
在函数上声明异常。便于提高安全性,让调用出进行处理,不处理编译失败
4、对多异常的处理
(1)、声明异常时,建议声明更为具体的异常,这样处理的可以更具体
(2)、对方声明几个异常,就对应几个catch块
如果多个catch块中的异常出现继承关系,父类异常catch块放在最下面
建议在进行catch处理方式,catch中一定要定义具体处理方式,也不要简单定义一条输出语句。用硬盘文件记录异常
class Demo
{
//数据角标越界异常,算术异常
public int div(int a,int b) throws ArithmeticException ,ArrayIndexOutOfBoundsException
{
int[] arr=new int[a];
System.out.println(arr[2]);
return a/b;
}
}
class ExceptionDemo
{
public static void main(String[] args)
{
Demo d=new Demo();
try
{
int x=d.div(3,0);
System.out.println("x="+x);
}
/*catch(Exception e) 不能加这里
{
System.out.println(" ... "+e.toString());
}*/
catch(ArithmeticException e)
{
System.out.println("除零了");
//System.out.println("BY ZErO");
System.out.println(e.getMessage()); //by zero
System.out.println(e.toString()); //异常名称:异常信息
e.printStackTrace(); //异常名称:异常信息,异常出现的位置
//其实jvm默认的异常处理机制,就是在调用printStackTrace()方法,打印异常的堆栈的跟踪信息
}
catch (ArrayIndexOutOfBoundsException e) //Exception e=new ArithmeticException();
{
System.out.println("数组角标越界异常");
e.printStackTrace();
}
/*catch(Exception e) 一般不用
{
System.out.println(" ... "+e.toString());
}*/
System.out.println("over");
//byte[] bt=new byte[1024*1024*800];
}
}
5、自定义异常
/**
因为项目中会出现特有的问题,而这些问题并未被Java所描述并封装对象
所以对于这些特有的问题可以按照Java的对问题封装的思想,将特有问题,进行自定义的异常封装
自定义异常
需求:在本程序中,对于除数是-1,也视为是错误的是无法进行运算。
那么就需要对这个问题进行自定义的描述。
*/
class FuShuException extends Exception
{
/*
一般情况下,函数内出现异常,函数上需要声明
发现打印的结果中只有异常的名称,却没有异常的信息。因为自定义的异常并为定义信息。
如何定义异常信息呢。
private String message;
FuShuException(String message)
{
this.message=message;
}
public String getMessage()
{
return message;
}*/
/*因为父类中已经把异常信息的操作都完成了,
所以子类只要在构造时,将异常信息传递给父类通过super语句
那么就可以直接通过getMessage方法获取自定义的异常信息。*/
private int value;
public
FuShuException(String message)
{
}
FuShuException(String message,int value)
{
super(message);
this.value=value;
}
public int getValue()
{
return value;
}
}
class Demo
{
public int div(int a,int b) throws FuShuException
{
if(b<0)
throw new FuShuException("除数为负"); //当在函数内部出现了throw抛出异常对象时,那么就必须要给对应的处理动作,要么在内部try catch处理,要么在函数上
//声明让调用者处理
return a/b;
}
}
class ExceptionDemo
{
public static void main(String[] args)
{
Demo d=new Demo();
try
{
d.div(4,-2);
}
catch (FuShuException e)
{
System.out.println(e.toString());
System.out.println("you are win");
}
System.out.println("over");
}
}
自定义异常:
必须是自定义类继承Exception。
为什么要继承Exception原因:
异常体系有一个特点,因为异常类和异常对象都需要被抛出。
他们都具备可抛性,这个可抛性是throwable这个体系中独有特点
只有这个体系中的类和对象才能被throw和throws操作。
throw和throws的区别
throws使用在函数上
throw使用在函数内
throws后面跟的是异常类,可以跟多个
throw后面跟的是异常对象
class Demo
{
public int div(int a,int b) throws ArithmeticException
{
//if(b==0)
// throw new ArithmeticException("除数是零了"); 换成new Exception("除数是零了");
return a/b;
}
}
class RuntimeExceptionDemo
{
public static void main(String[] args)
{
Demo d=new Demo();
d.div(4,0);
}
}
Exception中有个特殊的子类异常RuntimeException运行异常,
如果在函数内容抛出该异常,函数上可以不用声明,编译一样通过
如果在函数上声明了该异常,调用者可以不用进行处理,编译一样通过
之所以不用再函数声明,是因为不需要让调用者
当该异常发生,希望程序停止,因为在运行时,出现了无法继续运算的情况,希望停止程序后,对代码进行修正。
自定义异常时,如果该异常的发生,无法再继续进行运算,就让自定义异常继承RuntimeException
对于异常分两种
1、编译时被检测的异常
2、编译时不被检测的异常(运行时异常,RuntimeException及其子类)
异常格式:
1、第一种格式
try
{}
catch
{}
2、第二种格式
try
{
}
catch()
{}
finally
{}
3、第三种格式
try
{}
finally
{
一定会被处理//关资源
}
注意:catch使用于处理异常的,如果没有catch就代表异常没有被处理,如果该异常是检测时异常,那么必须声明
/*
异常在子父类覆盖中的体现
1、子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常
或者该异常的子类
2、如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子类
3、如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方法时,也不可以抛出异常
如果子类方法中发生了异常,就必须要进行try处理,绝对不能抛。
*/
如果函数声明了异常,调用者需要进行处理。处理方式可以throws可以try。
|
|