• 面向对象之异常


       异常

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

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

       对于问题的划分:两种:一种是严重的问题,一种是非严重的问题。

    1. 对于严重的,java通过Error类进行描述。对于Error一般不编写针对性的代码对其进行处理
    2. 对于非严重的,java通过Exception类进行描述。对于Exception可以使用针对性的处理方式进行处理

       无论Error或者Exception都具有一些共性内容,比如:不正常情况的信息,引发原因等。

       Throwable

             |-----Error

                 ------XXXX

             |-----Exception

                ------XXXX

       异常的处理

       java提供了特有的语句进行处理

       格式:

    try {
        需要被检测的代码;
     } catch(异常类   变量) {
        处理异常的代码;(处理方式)
     } finally {
        一定会执行的语句;
     }

       对捕获到的异常对象进行常见方法操作:

    1. String getMessage();获取异常信息。
    2. String toString();获取异常名称:异常信息
    3. void printStackTrace();获取异常名称,异常信息,异常出现的位置,其实JVM默认的异常处理机制,就是在调用printStackTrace(),打印异常在堆栈中的跟踪信息

       示例代码如下:

    class Demo {
        int div(int a, int b) {
            return a/b;  //new ArithmeticException();
        }
    }
    public class ExceptionDemo {
    
        public static void main(String[] args) {
            Demo d = new Demo();
            try {
                int x = d.div(4, 0); //new ArithmeticException();
                System.out.println("x="+x);
            } catch(Exception e) { //Exception e = new ArithmeticException();
                System.out.println("除零啦");
                System.out.println(e.getMessage()); // / by zero
                System.out.println(e);//异常名称:异常信息
                e.printStackTrace();//异常名称,异常信息,异常出现的位置
                                    //其实JVM默认的异常处理机制,就是在调用printStackTrace(),打印异常在堆栈中的跟踪信息
            }
            
            System.out.println("over");
        }
    
    }

       在函数上声明异常(throws)。便于提高安全性,让调用者进行处理,不处理编译失败。

       如下:

    class Demo {
        int div(int a, int b) throws Exception {//在功能上通过throws的关键字声明了该功能有可能会出现问题
            return a/b;  //new ArithmeticException();
        }
    }
    public class ExceptionDemo {
    
        public static void main(String[] args) {
            Demo d = new Demo();
            try {
                int x = d.div(4, 0);//new ArithmeticException();
                System.out.println("x="+x);
            } catch (Exception e) {
    //            e.printStackTrace();
                System.out.println(e.toString());
            } 
            
            System.out.println("over");
        }
    
    }

        对多异常的处理

    1. 声明异常时,建议声明更为具体的异常,这样处理的可以更具体
    2. 对方声明几个异常,就对应有几个catch块,不要定义多余的catch块。如果多个catch块中的异常出现继承关系,父类异常catch块放在最后

       建议在进行catch处理时,catch中一定要定义具体处理方式。不要简单定义一句:e.printStackTrace();也不要简单的就书写一条输出语句。

       示例如下:

    class Demo {
        int div(int a, int b) throws ArithmeticException, ArrayIndexOutOfBoundsException {//在功能上通过throws的关键字声明了该功能有可能会出现问题
            
            int[] arr = new int[a];
            System.out.println(arr[4]);
            return a/b;  //new ArithmeticException();
        }
    }
    public class ExceptionDemo {
    
        public static void main(String[] args) {
            Demo d = new Demo();
            try {
                int x = d.div(5, 0);//new ArithmeticException();
                System.out.println("x="+x);
            } catch (ArithmeticException e) {
    //            e.printStackTrace();
                System.out.println(e.toString());
                System.out.println("除零了");
            } catch (ArrayIndexOutOfBoundsException e) {
    //            e.printStackTrace();
                System.out.println(e.toString());
                System.out.println("角标越界了");
            } catch(Exception e) {
                System.out.println("haha:"+e.toString());
            } 
            
            System.out.println("over");
        }
    
    }

       自定义异常

       因为项目中会出现特有的问题,而这些问题并未被java所描述并封装对象。 所以对于这些特有的问题可以按照java的对问题封装的思想,将特有的问题,进行自定义的异常封装。

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

       发现打印的结果中只有异常的名称,却没有异常的信息,因为自定义的异常并未定义信息。

       如何定义异常信息?

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

       自定义异常,必须是自定义类继承Exception。

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

       示例代码如下:

    /*
     * 需求:在本程序中,对于除数是-1,也视为是错误的,是无法进行运算的,那么就需要对这个问题进行自定义的描述
     */
    class FuShuException extends Exception {
        private int value;
        
        public FuShuException() {
            super();
        }
        
        public FuShuException(String msg, int value) {
            super(msg);
            this.value = value;
        }
        
        public int getValue() {
            return value;
        }
    }
    class Demo1 {
        int div(int a, int b) throws FuShuException {
            if(b < 0) {
                throw new FuShuException("出现了除数是负数的情况------ / by fushu", b); //手动通过throw关键字抛出一个自定义异常对象
            }
            return a/b;
        }
    }
    public class ExceptionDemo1 {
    
        public static void main(String[] args) {
            Demo1 d = new Demo1();
            try {
                int x = d.div(4, -1);
                System.out.println("x="+x);
            } catch (FuShuException e) {
                System.out.println(e.toString());
    //          System.out.println("除数出现负数");
                System.out.println("错误的负数:"+e.getValue());
            }
            System.out.println("over");
        }
    
    }
    /*
    class Throwable {
        private String message;
        public Throwable(String message) {
            this.message = message;
        }
        public String getMessage() {
            return message;
        }
    }
    class Exception extends Throwable {
        Exception(String message) {
            super(message);
        }
    }
    new Exception("haha").getMessage();
    */

       throws和throw的区别:

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

       RuntimeException

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

    1. 如果在函数内容抛出该异常,函数上可以不用声明,编译一样通过。之所以不用在函数上声明,是因为不需要让调用者处理,当该异常发生,希望程序停止。因为在运行时,出现了无法继续运算的情况,希望停止程序后,希望程序员对代码进行修正。
    2. 如果在函数上声明了该异常,调用者可以不用进行处理,编译一样通过。

       自定义异常时:如果该异常的发生,无法再继续进行运算,就让自定义异常继承RuntimeException。

       对于异常分两种:

    1. 编译时被检测的异常
    2. 编译时不被检测的异常(运行时异常,RuntimeException以及其子类)

       示例代码如下:

    class FuShuException2 extends RuntimeException {
        FuShuException2(String msg) {
            super(msg);
        }
    }
    class Demo2 {
        int div(int a, int b) {
            if(b < 0) 
                throw new FuShuException2("出现了负数为负");
            if(b == 0)
                throw new ArithmeticException("被零除啦");
            return a/b;
        }
    }
    public class ExceptionDemo2 {
    
        public static void main(String[] args) {
            Demo2 d = new Demo2();
            int x = d.div(4, -9);
            System.out.println("x="+x);
            System.out.println("over");
        }
    
    }

       练习:毕老师上课。

       分析:开始思考上课中出现的问题。比如问题是:电脑蓝屏、电脑冒烟,要对问题进行描述,封装成对象。可是当冒烟发生后,出现讲课进度无法继续,出现了讲师的问题,课时计划无法完成。

       代码如下:(小小问题也能分析出这么多代码,要学习一个)

    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 = 3;//正常状态:1
        
        public void run() throws LanPingException, MaoYanException {
            if(state == 2) {
                throw new LanPingException("蓝屏了");
            }
            if(state == 3) {
                throw new MaoYanException("冒烟了");
            }
            System.out.println("电脑运行");
        }
        public void reset() {
            state = 1;
            System.out.println("电脑重启");
        }
    }
    class Teacher {
        private String name;
        private Computer cmpt;
        
        Teacher(String name) {
            this.name = name;
            cmpt = new Computer();
        }
        public void prelect() throws NoPlanException {
            try {
                cmpt.run();
            } catch (LanPingException e) {
                cmpt.reset();
            } catch (MaoYanException e) {
                test();
                throw new NoPlanException("课时无法继续"+e.getMessage());
            }
            System.out.println("讲课");
        }
        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("换老师,或者放假");
            }
        }
    
    }

       

       finally

       finally代码块:定义一定执行的代码。通常用于关闭资源。

       格式:

       1、try {

           } catch() {

           } 

       2、try {

           } catch() {

           } finally {

           }

       3、 try {

            } finally {

            }

       finally使用举例:

       1、

    class FuShuException3 extends Exception {
        FuShuException3(String msg) {
            super(msg);
        }
    }
    class Demo3 {
        int div(int a, int b) throws FuShuException3 {
            if(b < 0) 
                throw new FuShuException3("出现了除数为负");
            return a/b;
        }
    }
    public class ExceptionDemo3 {
    
        public static void main(String[] args) {
            Demo3 d = new Demo3();
            try {
                int x = d.div(4, -1);
                System.out.println("x="+x);
            } catch(FuShuException3 e) {
                System.out.println(e.toString());
                return;
            } finally {
                System.out.println("finally"); //finally中存放的是一定会被执行的代码
            }
            System.out.println("over");
        }
    
    }

       2、操作数据库

    class NoException extends Exception {
        
    }
    public void method() throws NoException {
        连接数据库;
        
        数据操作; //throw new SQLException();
        
        关闭数据库;//该动作,无论数据操作是否成功,一定要关闭资源。
        
        try {
            连接数据库;
            
            数据操作; //throw new SQLException();
        } catch(SQLException e) {
            会对数据库进行异常处理;
            throw new NoException();
        } finally {
            关闭数据库;
        }
        
    }

       记住一点:catch是用于处理异常,如果没有catch就代表异常没有被处理过,如果该异常是检测时异常,那么必须声明。

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

    1. 子类在覆盖父类时,如果父类的方法抛出异常,那么子类的覆盖方法,只能抛出父类的异常或者改异常的子类或者不抛
    2. 如果父类方法抛出多个异常,那么子类在覆盖多个方法时,只能抛出父类方法的子集(不抛也行)
    3. 如果父类或者接口的方法中没有异常抛出,那么子类在覆盖方法时,也不可以抛出异常。如果子类方法发生了异常,就必须要进行try处理,绝对不能抛。

       示例代码如下:

    class AException extends Exception {
        
    }
    class BException extends AException {
        
    }
    class CException extends Exception {
        
    }
    /*
     Exception
        |---AException
              |---BException
        |---CException
     */
    class Fu {
        public void show() throws AException {
    
        }
    }
    class Test {
        public void function(Fu f) {
            try {
                f.show();
            } catch (AException e) {
                e.printStackTrace();
            }
        }
    }
    class Zi extends Fu {
        public void show() throws AException { //抛CException编译失败
            
        }
    }

       练习:

       有一个圆形和长方形,它们都可以获取面积,对于面积如果出现非法的数值,视为是获取面积出现问题。问题通过异常来表示。

       代码如下:

    class NoValueException extends RuntimeException {
        NoValueException(String msg) {
            super(msg);
        }
    }
    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("出现非法值");
            }
            this.len = len;
            this.wid = wid;
        }
        
        public void getArea() {
            
            System.out.println(len*wid);
        }
    }
    
    class Circle implements Shape {
        private int radius;
        public static final double PI = 3.14; 
        
        Circle(int radius) {
            if(radius <= 0) {
    //            throw new RuntimeException("非法半径");//可以直接抛RuntimeException
                throw new NoValueException("非法半径");
            }
            this.radius = radius;
        }
        
        public void getArea() {
            System.out.println(radius*radius*PI);
        }
    }
    public class ExceptionTest1 {
    
        public static void main(String[] args) {
    //        Rec r = new Rec(-3, 4);
    //        r.getArea();
            
            Circle c = new Circle(-8);
            c.getArea();
            System.out.println("over");
            
        }
    
    }

     


       总结:

       异常:对问题的描述,将问题进行对象的封装。


       异常体系:

       Throwable
           |---Error
           |---Exception
                   |---RuntimeException

       异常体系的特点:异常体系中的所有类以及建立的对象都具有可抛性,也就是说可以被throw和throws关键字所操作。只有异常体系具备这个特点。


       throw和throws的用法:

       throw定义在函数内,用于抛出异常对象。

       throws定义在函数上,用于抛出异常类,可以抛出多个,用逗号隔开。

       当函数内容有throw抛出异常对象,并未进行try处理,必须要在函数上声明,否则编译失败。

       注意:RuntimeException除外,也就是说,函数内如果抛出的是RuntimeException异常,函数上可以不用声明。

       如果函数声明了异常,调用者需要进行处理,处理方式可以throws也可以try。


       异常有两种:

       编译时被检测异常

             该异常在编译时,如果没有处理(没有抛也没有try),编译失败

             该异常被标识,代表这可以被处理

       运行时异常(编译时不检测)

             在编译时,不需要处理,编译器不检查

             该异常的发生,建议不处理,让程序停止,需要对代码进行修正


       异常处理语句

    try {
        需要被检测的代码;
    } catch() {
        处理异常的代码;
    } finally {
        一定会指执行的代码;
    }

       有三种结合格式:

       1、

    try {
        需要被检测的代码;
    } catch() {
        处理异常的代码;
    }

       2、

    try {
        需要被检测的代码;
    } finally {
        一定会指执行的代码;
    }

       3、

    try {
        需要被检测的代码;
    } catch() {
        处理异常的代码;
    } finally {
        一定会指执行的代码;
    }

       注意:

       1、finally中定义的通常是关闭资源代码,因为资源必须释放。

       2、finally只有一种情况不会执行,当执行到System.exit(0);finally不会被执行。


       自定义异常:定义类继承Exception或者RuntimeException

    1. 为了让自定义类具备可抛性
    2. 让该类具备操作异常的共性方法

       当要定义自定义异常的信息时,可以使用父类已经定义好的功能。将异常信息传递给父类的构造函数。

    class MyException extends Exception {
        MyException(String message) {
            super(message);
        }
    }

       自定义异常:按照java的面向对象思想,将程序出现的特有问题进行封装。


       异常的好处:

    1. 将问题进行封装
    2. 将正常流程代码和问题处理代码相分离,方便于阅读

       异常的处理原则:

    1.  处理方式有两种:try或者throws
    2. 调用到抛出异常的功能时,抛出几个,就处理几个。一个try对应多个catch。
    3. 多个catch,父类的catch放到最下面
    4. catch内,需要定义针对性的处理方式,不要简单的定义printStackTrace或者输出语句。
      也不要不写。

       当捕获到的异常,本功能处理不了时,可以继续在catch中抛出

    try{
        throw new AException();
    } catch(AException e) {
        throw e;
    }

        如果该异常处理不了,但并不属于该功能出现的异常。可以将异常转换后,在抛出和该功能相关的异常。或者异常可以处理,但需要将异常产生后,和本功能相关的问题提供出去,让调用者知道并处理,  也可以将捕获异常处理后,转换新的异常抛出。(比如,汇款的例子)

    try{
          throw new AException();
    } catch(AException e) {
          //对AException处理
        throw new BException();
    }

       异常的注意事项:

       在子父类覆盖时

    1. 子类抛出的异常必须是父类的异常的子类或者子集
    2. 如果父类或者接口没有异常抛出时,子类覆盖出现异常,只能try不能抛。

       

       面试时应注意的一个小问题:

       1、

    /*
    输出:
    B
    C
    D
     */
    public class DemoTest {
    
        public static void main(String[] args) {
            try {
                showExce();
                System.out.println("A");
            } catch(Exception e) {
                System.out.println("B");
            } finally {
                System.out.println("C");
            }
            System.out.println("D");
        }
        public static void showExce() throws Exception { //声明,这个方法有可能会出问题,有可能不会出问题
            throw new Exception();//把抛出异常封装 在方法之中,外面不知道
        }
        
    }

       与之相反的另一个题目是:

    /*
    编译失败。因为打印“A”的输出语句执行不到,就是一句废话(return语句后面也不能写代码一个意思)
    记住:throw单独存在时,下面不要定义语句。因为执行不到
     */
    class Demow {
        public static void func() {
            try {
                throw new Exception();
                System.out.println("A");
            } catch(Exception e) {
                System.out.println("B");
            }
        }
        public static void main(String[] args) {
            try {
                func();
            } catch(Exception e) {
                System.out.println("C");
            }
            System.out.println("D");
        }
    }

       2、一道java国际考试题

    public class DemoTest {
        
        private static String output = "";
        public static void foo(int i) {
            try {
                if(i == 1)
                    throw new Exception();
                output+="1";
            } catch(Exception e) {
                output+="2";
                return;
            } finally {
                output+="3";
            }
            output+="4";
        }
        
        public static void main(String[] args) {
            foo(0);
            System.out.println(output);//134
            foo(1);
            System.out.println(output);//13423
        }
        
    }
  • 相关阅读:
    document基本操作 动态脚本-动态样式-创建表格
    js原型模式和继承
    NuGet本地包自定义路径
    技术文档链接收藏
    数据结构排序
    Insertion Sort
    选择排序之javascript
    冒泡排序之javascript
    C++双向链表
    单向链表
  • 原文地址:https://www.cnblogs.com/yerenyuan/p/5207777.html
Copyright © 2020-2023  润新知