Java基础:异常

一.异常继承体系

异常是程序运行过程中出现的错误。Java把异常当作对象来处理,把异常信息封装成了一个类,并定义一个基类java.lang.Throwable作为所有异常的超类。

Throwable:他是所有错误和异常的超类,有两个子类Error和Exception

Error:错误。程序无法处理的错误,比如OutOfMemoryError、ThreadDeath等。这些异常发生时,Java虚拟机(jvm)一般会选择线程终止。

Exception:异常。程序本身可以处理的异常,程序中应当尽可能去处理这些异常。

RuntimeException:运行期异常,Java程序运行过程中出现的问题

CheckableException:编译时异常或可检查异常,是除了RuntimeExecption之外所有的Exception

运行时异常:

方法中抛出运行时期异常,方法定义中无需throws声明,调用者也无需处理此异常

运行时异常一旦发生,需要程序人员修改源代码

编译时异常:

必须在编译前处理,否则无法通过编译

二.发生异常时,程序的执行特征:

从发生异常的地方,开始被一分为二

在异常发生之前的代码,都可以正常运行,之后的代码不会运行

当异常发生的时候,这个异常被jvm捕获,并将这个异常的相关信息,创建一个异常对象,然后将该对象的信息输出到控制台(执行的是虚拟机默认的异常处理代码)

终止当前程序

三.异常与错误的区别

异常:指程序在编译运行期间发生了某种异常(Exception),我们可以对异常进行具体的处理,若不处理异常,程序将会结束运行。

错误:指程序在运行期间发生了某种错误(Error),Error错误通常没有具体的处理方式,程序将会运行结束,Error错误的发生往往都是系统级别的问题,都是jvm所在的系统发生并反馈给居民、的。我们只能修改代码。

四.抛出异常throw

在Java中,提供了一个throw关键字,它用来抛出一个指定的(Throwable类型)异常对象

一般用于程序出现某种逻辑时,程序员主动抛出某种特定类型的异常

步骤:创建一个异常对象,封装一些提示信息(信息可以自己填写)

通过关键字throw,将这个异常告知调用者

throw用在方法内,用来抛出一个异常对象,将这个异常对象传递到调用者处,并结束当前方法的运行

使用格式:throw new 异常类名(参数)

throw new NullPointerException("要访问的arr数组不存在");

throw new ArrayIndexOutOfBoundException("该索引在数组中不存在,已经超出范围");

//具体实例

public static void main(String[] args){

        String s="abc";

        if(s.equals("abc")){

                throw new NumberFormatException();

        }else{

                System.out.println(s);

        }

}

注意:

如果抛出的异常对象属于可检查的异常,必须在该方法头部,声明抛出异常,即throws要抛出的异常类型

其次,对于抛出可检查的异常,还必须与方法的异常列表中的异常兼容

如果父类方法声明了异常列表:

子类可以不声明异常列表

子类方法有自己的异常列表时,必须保证子类的异常列表中包含的异常类型,与父类中包含的异常类型兼容

五.声明异常throws

声明:将问题标识出来,报告给调用者

throws是方法可能抛出异常的声明,如果定义功能时有问题发生需要报告给调用者。可以通过在方法上使用throws关键字进行声明

对于声明了会抛出可检查异常的方法,就意味着这个方法会产生可检查异常,所以一旦调用就必须对该方法进行异常处理

throws用于进行异常类的声明,若该方法可能有多种异常情况产生,那么在throws后面可以写多个异常类,用逗号隔开

声明异常格式:修饰符 返回值类型 方法名(参数)throws<异常列表>{}

public static void function() throws NumberFormatException{

        String s="abc";

        System.out.println(Double.parseDouble(s));

}

public static void main(String[] args){

        try{

                function();

        }catch(NumberFormatException e){

                System.err.println("非数据类型不能转换");

        }

}

六.捕获异常try catch finally

try{

        //需要被检测的语句

}

catch(异常类 e){ //try中抛出的什么异常,在括号里就定义什么异常类型

        //异常的处理语句        

}

finally{

        //一定会被执行的语句

}

异常处理流程:

首先,当异常在try代码块中发生的时候,虚拟机首先捕获这个异常,创建一个异常对象(包含本次异常的所有详细信息)

虚拟机会把这个异常抛给catch代码块(方法调用catch中处理异常的代码块)

执行catch代码块中处理异常的代码

没有终止我们的应用程序,而是从catch语句之后的代码开始,继续执行我们的应用程序

七.try catch finally异常处理的组合方式

检测异常,并传递给catch处理,并在finally中进行资源释放

即使在finally代码块中执行了return语句,finally代码块中的代码仍然会执行

多个try catch组合可以处理可能发生的多种不同类型的异常

实际开发的时候:

try中的代码尽可能少

将相关的异常放在同一个try中

相关异常指一个连续流程中,可能发生的一系列异常

若catch中多个异常之间有父子类异常的关系,那么先写子类异常类型,再写父类异常类型

当发生异常时,最多执行一个catch分支的代码

public class Demo{

        public static void main(String[] args){

                try{

                        int i=10;

                        int j=i/0;

                        System.out.println("try after exception");

                        //空指针异常

                        int[] a=null;

                        System.out.println(a[0]);

                        //数据越界异常

                        int [] b={1,2,3}; 

                        System.out.println(b[3]);

                }catch(ArithmeticException e){

                        System.out.println("发生了除零异常");

                }

                catch(NullPointerException|ArrayIndexOutOfBoundsException e){

                        System.out.println("发生了数组异常");

                }

        }             

}

八.异常在方法中重写中细节

子类覆盖父类方法,如果父类的方法声明异常,子类只能声明父类异常或者该异常的子类,声明异常也要继承于父类。

class Fu{

        public void method () throws RuntimeException{

        }

}

class Zi extends Fu{

        public void method() throws RuntimeException{}

        //public void method() throws NullPointerException{}

 }

当父类方法声明多个异常时,子类覆盖时只能能声明多个异常的子集

class Fu {
    public void method () throws NullPointerException, ClassCastException{
    }
}
class Zi extends Fu {
    public void method()throws NullPointerException, ClassCastException { }
    public void method() throws NullPointerException{ } //抛出父类异常中的一部分
    public void method() throws ClassCastException { } //抛出父类异常中的一部分
}

九、Throwable类中的常用方法

  • getCause():返回抛出异常的原因,即异常提示信息。如果 cause 不存在或未知,则返回 null。
  • getMessage():返回异常的消息信息,即该异常的名称与详细信息字符串
  • printStackTrace():在控制台输出该异常的名称与详细信息字符串、异常出现的代码位置

十、try和finally中都有return语句,执行哪一个return?

  • 首先要确定的一点是,不管有木有出现异常,finally块中代码都会执行
  • 当try和catch中有return时,finally仍然会执行;
  • finally是在return后面的表达式运算后执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,不管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的;
  • finally中最好不要包含return,否则程序会提前退出,返回值不是try或catch中保存的返回值。
问:如果try和finally语句里面都有return,会执行哪一个呢?
首先,在程序没有异常的情况下,首先执行到try里面的语句, 但是只执行到了return里面的****expression,expression首先存放在操作数栈顶,然后复制到局部变量区,并没有执行返回语句return(执行返回语句通常意味着程序执行结束)。然后执行finally,当执行到finally里面的return时候,会将return语句执行完整,此时程序已经有了返回值,因为,执行结束。
  • 总结:执行try块,执行到return语句时,先执行return的语句,但是不返回到main 方法,接下来执行finally块,遇到finally块中的return语句,执行,并将值返回到main方法,这里就不会再回去返回try块中计算得到的值

十一、自定义异常

  • 如果Java没有提供你需要的异常,则可以自定义异常类。
  • 编译时异常继承 Exception,运行时异常继承 RuntimeException
  • 格式:
Class 异常名 extends Exception{ //或继承RuntimeException
    public 异常名(){
    }
    public 异常名(String s){
        super(s);
    }
}
  • 示例:
需求描述:
定义Person类,包含name与age两个成员变量。
在Person类的有参数构造方法中,进行年龄范围的判断,若年龄为负数或大于200岁,则抛出NoAgeException异常,异常提示信息“年龄数值非法”。
要求:在测试类中,调用有参数构造方法,完成Person对象创建,并进行异常的处理。
//自定义异常类
class NoAgeException extends Exception{
    NoAgeException() {
        super();
    }

    NoAgeException(String message)  {
        super(message);
    }
}

//Person类
class Person{
    private String name;
    private int age;
    Person(String name,int age) throws NoAgeException   {
        //加入逻辑判断
        if(age<0 || age>200)        {
            throw new NoAgeException(age+",年龄数值非法");
        }
        this.name = name;
        this.age = age;
    }
    //定义Person对象对应的字符串表现形式。覆盖Object中的toString方法。
    public String toString()    {
        return "Person[name="+name+",age="+age+"]";
    }
}

//测试类
class ExceptionDemo{
    public static void main(String[] args) {
        try {
            Person p = new Person("xiaoming",20);
            System.out.println(p);
        }
        catch (NoAgeException ex){
            System.out.println("年龄异常啦");
        }
        System.out.println("over");
    }
}

总结一下,构造函数到底抛出这个NoAgeException是继承Exception呢?还是继承RuntimeException呢?
继承Exception,必须要throws声明,一声明就告知调用者进行捕获,一旦问题处理了调用者的程序会继续执行。
继承RuntimeExcpetion,不需要throws声明的,这时调用是不需要编写捕获代码的,因为调用根本就不知道有问题。一旦发生NoAgeException,调用者程序会停掉,并有jvm将信息显示到屏幕,让调用者看到问题,修正代码。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值