Java:Exception

异常:

就是程序在运行时出现不正常的情况。

异常的由来:问题也是现实生活中一个具体的事物,也可以通过java的类的形式进行描述,并封装成对象。其实就是Java对不正常情况进行描述后的对象的体现。

两种问题:

严重问题(Error)、非严重问题(Exception)。

Error和Exception具有一些共性内容。(都是Throwable的子类)

try
{
    需要被检测的代码
}
catch(异常类 变量)
{
    处理异常的代码;(处理方式)    
}

严重问题:

Java通过Error类进行描述。对于Error一般不编写针对性的代码对其进行处理。

非严重问题:

Java通过Exception类进行描述。对Exception可以使用针对性的代码进行处理。

异常的处理:

try
{
    需要被检测的代码
}
catch(异常类 变量)
{
    处理异常的代码;(处理方式)    
}

异常的常见操作:

getMessage() :返回信息:异常原因

toString():反馈信息:异常类+异常原因

printStackTrace():反馈堆栈中的异常信息 异常类+异常原因+异常出现的位置,其实jvm默认的异常处理机制就是在调用printStackTrace方法

例如,除数是零

class Demo
{
    int div(int a, int b)
    {
        return a/b;
    }
}

class ExceptionDemo 
{
    public static void main(String[] args) 
    {
        Demo d = new Demo();
        try
        {
            int x = d.div(4,0);//捕获到异常对象 new AritchmeticException()
            System.out.println("x=" + x);
        }
        catch (Exception e)//把捕获到的异常给了catch: Exception e = new AritchmeticException();
        {
            System.out.println("除数是零!");
            System.out.println(e.getMessage());
            System.out.println(e.toString());
            e.printStackTrace();
        }

        System.out.println("Program Over!");
    }
}

      但我们在定义Class时就应该提示是否会出现异常,提醒调用该Class时是需要处理Exception。当调用者使用该Class时,必须捕获该类的异常进行处理或者抛出。

捕获:try-catch处理

class Demo
{
    int div(int a, int b)throws Exception//在功能上通过throws关键字声明了该功能在使用时有可能出现问题
    {
        ...
    }
}

class ExceptionDemo 
{
    public static void main(String[] args) 
    {
        Demo d = new Demo();
        try
        {
            ...
        }
        catch (Exception e)
        {
            ...
        }
    }
}

抛出:调用该有标识异常的类的方法上通过throws抛出,不处理

class Demo
{
    int div(int a, int b)throws Exception//在功能上通过throws关键字声明了该功能在使用时有可能出现问题
    {
        ...
    }
}

class ExceptionDemo 
{
    public static void main(String[] args)throws Exception
    {
        ...
    }
}

·声明异常时,建议声明更为具体的异常,这样处理的可以更具体。

例如上面的例子,可以把异常由Exception更改为ArithmeticExcption

·有多个异常时,可以针对性用多个catch处理。如果多个异常有继承关系,父类异常的catch放在最下面

建议进行catch处理时,catch中一定要定义具体的处理方式,不要简单的使用printStackTrace()等等

class Demo
{
    int div(int a, int b)throws ArithmeticException,ArrayIndexOutOfBoundsException//抛出多个异常
    {

        int[] arr = new int[a];
        System.out.println(arr[5]);//可能出现异常的地方

        return a/b;//可能出现异常的地方
    }
}

class ExceptionDemo 
{
    public static void main(String[] args)
    {
        Demo d = new Demo();
        try
        {
            int x = d.div(4,0);
            System.out.println("x=" + x);
        }
        catch (ArithmeticException e)//处理异常
        {
            System.out.println("除数是零!");
            e.printStackTrace();
        }
        catch (ArrayIndexOutOfBoundsException e)//处理异常
        {
            System.out.println("数组角标异常!");
            e.printStackTrace();
        }

        System.out.println("Program Over!");
    }
}

自定义异常:

  因为项目中会出现特有问题,而这些问题是Java未对其描述并封装成对象。所以对这些特有问题可以按照Java对问题的封装思想,对其封装。

  当在函数内部出现了throw抛出异常对象,那么就必须要给出对应的处理。要么在内部try-catch,要么在函数上声明让调用者处理。 一般情况下,函数内出现异常,函数上需要声明。

如何自定义异常信息呢?
  因为父类已经把异常信息的操作都完成了,所以子类只要在构造时,将异常信息传递给父类,通过super语句。那么就可以直接通过getMessage方法获取自定义的异常信息了。

继承Exception的原因
  异常体系都有一个特点:异常类和异常对象都被抛出。
  他们都具有可抛性,是Throwable这个体系的独有特点。只有这个体系中的类和对象才可以被throws和throw操作。

throws 和 throw 的区别:
  1,throws使用在函数上,throw使用在函数内
  2,throws后面跟的是异常类,可以跟多个,用逗号隔开;throw后跟的是异常对象

 1 /*
 2 需求:在本程序中,不允许除数为负数。
 3 因为该需求是特殊要求,所以要对这个问题进行自定义的描述;
 4 */
 5 class DefinedException extends Exception
 6 {
 7     private int value;
 8 
 9     DefinedException()
10     {
11         System.out.println("自定义异常的空参数的构造函数:除数不能为负数");
12     }
13     DefinedException(String msg)
14     {
15         super(msg);
16     }
17     DefinedException(String msg, int value)
18     {
19         super(msg);
20         this.value = value;
21     }
22 
23     public int getValue()
24     {
25         return value;
26     }
27 }
28 
29 class Demo
30 {
31     int div(int a,int b)throws DefinedException
32     {
33         if (b<0)
34         {
35             //throw new DefinedException();
36             //throw new DefinedException("除数不能为负数!");
37             throw new DefinedException("除数不能为负数!",b);//手动通过throw抛出
38         }
39         return a/b;
40     }
41 }
42 class DefinedExceptionDemo 
43 {
44     public static void main(String[] args) 
45     {
46         Demo d = new Demo();
47         try
48         {
49             int x = d.div(4,-1);
50             System.out.println("a/b="+x);
51         }
52         catch (DefinedException e)
53         {
54             System.out.println("除数不能为负数!");
55             System.out.println("错误的数是:"+e.getValue());
56             //d.printStackTrace();
57         }
58 
59         System.out.println("Program Over!");
60     }
61     
62 }

 RuntimeException:

  Exception中有一个特殊的子类异常RuntimeException 运行时异常。

  RuntimeException或者其子类的异常,可以不用在函数上使用throws申明。如果在函数上申明了该异常,调用者可以不用处理就可以通过编译。不申明的原因:当程序运行时遇到该异常,导致下面的程序不能预期运行,希望停止运行后修正该程序。(比如如果一个函数只申明了一个异常,而在函数内抛出了三个异常,那么另外两个异常必然就是RuntimeException或者其子类)


  自定义异常时,如果该异常发生后无法继续工作,就让该异常继承RuntimeException

异常分两种:
  1,编译时被检测的异常
  2,编译时不被检测的异常,但运行时异常。RuntimeException及其子类

finally:

  finally中的代码是一定会执行的代码,一般存放必须要操作的内容比如释放与数据库的连接、关闭IO等等。

  finally只有一种情况不会被执行到:finally前面有System.exit(0)

异常处理的三种各式:

1, try-catch

2, try-catch-finally

3,   try-fiinally

异常在子父类覆盖时的体现:

1,子类在覆盖父类方法时,如果父类的方法抛出异常,那么子类的覆盖方法只能抛出父类的异常或者该异常的子类

2,如果父类方法抛出多个异常,那么子类在覆盖该方法时,只能抛出父类异常的子集

3,如果父类或者接口的方法中,没有异常抛出,那么子类在覆盖方法时也不可以抛出异常

  如果子类方法发生了异常,就必须要try处理,绝对不能抛。

原文地址:https://www.cnblogs.com/siyingcheng/p/4307858.html