概述
异常:是运行时期发生的不正常情况打。在java中用类的形式对不正常情况进行了描述和封装对象,描述不正常的情况的类,就称为异常类。以前正常流程代码和问题处理代码相结合,提高代码的阅读性。其实异常就是java通过面向对象的思想将问题封装成了对象。用异常类对其进行描述,不同的问题用不同的类进行具体描述,比如角标越界,空指针等等。
问题很多,意味着描述的类也很多,将其共性进行向上抽取,形成了异常体系。最终(不正常问题)就分成了两大类
Throwable:
无论是error,还是异常,问题,问题发生就应该抛出,让调用者知道并处理。
该体系的特点就在于Throwable及其所有的子类都具有可抛性。thorws,throw这两个关键字进行操作的类都具有可抛性。
- 一般不可处理问题。Error
特点:是由jvm抛出的严重性问题。这种问题发生一般不针对性处理,直接修改程序。
例子:java是用class作为运行文件,但运行文件不在,就处理不了。
- 2.可以处理问题。Exception
特点:子类的后缀名都是用其父类名作为后缀,阅读性很强
抛出
例子:
package com.monfolld;
class Demo2{
public int method(int[] arr,int index){
if(arr==null){
throw new NullPointerException("数组的引用不能为空");
}
if (index>=arr.length){
throw new ArrayIndexOutOfBoundsException("数组角标越界了,你是不是放了"+index);
}
if(index<0){
throw new ArrayIndexOutOfBoundsException("数组角标是负数,你是不是放了"+index);
}
return arr[index];
}
}
public class ExceptionDemo {
public static void main(String[] args){
int arr[]=new int[3];
Demo2 d=new Demo2();
int num=d.method(arr,3);
System.out.println("num"+num);
}
}
输出结果:
Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 数组角标越界了,你是不是放了3
at com.monfolld.Demo2.method(ExceptionDemo.java:5)
at com.monfolld.ExceptionDemo.main(ExceptionDemo.java:15)
对于角标是整数不存在,可以用角标越界表示。
对于负数角标的情况,准备用负数角标异常表示。但负数角标这种异常在java中并没有定义过,那就按照java异常的创建思想,将负数角标进行自定义描述,并封装成对象,这种自定义问题描述称为自定义异常。
注意:如果让一个类称为异常类,必须要继承异常体系,因为只有称为异常体系的子类才有资格具备可抛性。才可以被两个关键字操作,throws throw
package com.monfolld;
class FuShuIndexException extends Exception{
FuShuIndexException(){
}
FuShuIndexException(String a){
super(a); //找父类
}
}
class Demo3{
public int method(int[] arr,int index)throws FuShuIndexException
{
if(index<0){
throw new FuShuIndexException("数组角标是负数");
}
return arr[index];
}
}
public class ExceptionDemo2 {
public static void main(String[] args)throws FuShuIndexException
{
int arr[]=new int[3];
Demo3 d=new Demo3();
int num=d.method(arr,-3);
System.out.println("num"+num);
}}
输出结果:
Exception in thread "main" com.monfolld.FuShuIndexException: 数组角标是负数
at com.monfolld.Demo3.method(ExceptionDemo2.java:15)
at com.monfolld.ExceptionDemo2.main(ExceptionDemo2.java:25)
异常的分类:
1.编译时被检测异常:只要是Exception和其子类都是,除了特殊子类RuntimeException体系。
这种问题一旦出现,希望在编译时就进行检测,让这种问题有对应的处理方式。
2.编译时不检测异常(运行时异常):就是Exception中的RuntimeException自体系和其子类。
这种问题的发生,无法让功能继续,运算无法进行,更多是因为调用的原因导致的而或者引发了内部状态的改变导致的。那么这种问题一般不处理,直接编译通过,再运行时,让调用者调用的程序强制停止,让调用者对代码进行修正。
throw和throws区别:
1.throw使用在函数上
throws使用在函数内。
2.throw抛出的是异常类,可以抛多个,用逗号隔开
throws抛出的是异常对象。
捕捉:
异常处理的捕捉形式:这里是可以对异常进行针对性的处理方式。
具体格式:
try{
//需要被检测异常的代码
}
catch(异常类 变量){//该变量用于接收发生异常对象
//处理异常的代码
}
finally{ //通常用于(关闭)释放资源
//一定会被执行的代码
}
package com.monfolld;
class FuShuIndexException extends Exception{
FuShuIndexException(){
}
FuShuIndexException(String a){
super(a);
}
}
class Demo4{
public int method(int[] arr,int index) throws FuShuIndexException
{
if(index<0)
throw new FuShuIndexException("数组角标是负数");
return arr[index];
}
}
public class ExceptionDemo4 {
public static void main(String[] args){
int[] arr=new int[3];
Demo4 d=new Demo4();
try {
int num=d.method(arr,-30);
}
catch (FuShuIndexException e){
System.out.println("负数角标异常");
System.out.println("String"+e.toString()); //将对象编程字符串
System.out.println(e.getMessage()); //打印异常信息
e.printStackTrace(); //jvm的异常处理机制就是调用异常的这个方法
}
}
}
结果:
负数角标异常
Stringcom.monfolld.FuShuIndexException: 数组角标是负数
数组角标是负数
com.monfolld.FuShuIndexException: 数组角标是负数
at com.monfolld.Demo4.method(ExceptionDemo4.java:13)
at com.monfolld.ExceptionDemo4.main(ExceptionDemo4.java:22)
当一个try里面的异常对应多个catch需要处理时,父类的catch一定要放在最下面
package com.monfolld;
class FuShuIndexException extends Exception{
FuShuIndexException(){
}
FuShuIndexException(String a){
super(a);
}
}
class Demo4{
public int method(int[] arr,int index) throws FuShuIndexException,NullPointerException
{ if(arr==null)
throw new NullPointerException("没有任何数组实体");
if(index<0)
throw new FuShuIndexException("数组角标是负数");
return arr[index];
}
}
public class ExceptionDemo4 {
public static void main(String[] args){
int[] arr=new int[3];
Demo4 d=new Demo4();
try {
int num=d.method(null,-30);
}
catch (NullPointerException e){
System.out.println(e.toString());
}
catch (FuShuIndexException e){
System.out.println("负数角标异常");
System.out.println("String"+e.toString()); //将对象编程字符串
System.out.println(e.getMessage()); //打印异常信息
e.printStackTrace(); //jvm的异常处理机制就是调用异常的这个方法
}
catch (Exception e){ //这个catch一定要放在最下面
}
}
}
异常处理的原则
1.函数内容如果抛出需要检测的异常,那么函数上必须声明,否则必须在函数内用trycatch捕捉,否则编译失败。
2.如果调用到了声明异常的函数,要么try catch要么throws,否则编译失败。
3.什么时候catch,什么时候throws?
功能内容可以解决,用catch
解决不了,用throws告诉调用者,由调用者解决
4.一个功能如果抛出了多个异常,那么调用时,必须有对应多个catch进行针对性处理,内部有几个需要检测的异常,就抛几个异常,抛出几个,就catch几个。
异常的应用
老是用电脑上课。问题领域涉及两个对象:老师,电脑。
分析其中的问题,比如:1.电脑蓝屏;2.电脑冒烟。
package com.monfolld;
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 Computer{
private int state=2;//电脑的正常运行状态为0,蓝屏为1,冒烟为2
public void run()throws LanPingException,MaoYanException{ //预处理
System.out.println("电脑运行");
if (state==1){
throw new LanPingException("电脑蓝屏了");
}
if (state==2){
throw new MaoYanException("电脑冒烟了");
}
}
public void reset(){
state=0;
System.out.println("电脑重启");
}
}
class Teacher {
private String name;
private Computer comp;
Teacher(String name) {
this.name = name;
comp = new Computer();
}
public void prelect()throws NoPlanException{ //讲课
try {
comp.run();
System.out.println("讲课");
}
catch (LanPingException e) { //蓝屏的时候,重启继续讲课
comp.reset();
prelect();
}
catch (MaoYanException e) { //冒烟的时候,学生联系,
System.out.println(e.toString());
test();
throw new NoPlanException("课时进度无法完成"+e.getMessage());
}
}
public void test(){
System.out.println("大家练习");
}
}
public class ExceptionTest {
public static void main(String[] args){
Teacher t=new Teacher("老师");
try {
t.prelect();
}
catch (NoPlanException e){ //异常转换
System.out.println(e.toString()+".....");
System.out.println("换人");
}
}
}
异常处理的注意事项
1.子类在覆盖父类方法时,父类方法如果抛出了异常,那么子类的方法只能抛出父类的异常或者该异常的子类。
2.如果父类抛出多个异常,那么子类只能抛出父类异常的子集。
注意:如果父类没有抛出异常,那么子类覆盖时绝对不能抛,就只能try。