• java中的异常机制总结


    1、异常分类

    Java是采用面向对象的方式来处理异常的。处理过程:

    1. 抛出异常:在执行一个方法时,如果发生异常,则这个方法生成代表该异常的一个对象,停止当前执行路径,并把异常对象提交给JRE。

    2. 捕获异常:JRE得到该异常后,寻找相应的代码来处理该异常。JRE在方法的调用栈中查找,从生成异常方法开始回溯,直到找到相应的异常处理代码为止。

    try {

            copyFile("d:/a.txt","e:/a.txt");

        } catch (Exception e) {

            e.printStackTrace();

        }

    Java对异常进行了分类,不同类型的异常分别用不同的Java类表示,所有异常的根类为java.lang.Throwable,Throwable下面又派生了两个子类:Error和Exception。Java异常类的层次结构如图6-2所示。

     

    图1-1 Java异常类层次结构图

    1、Error表明系统JVM已经处于不可恢复的崩溃状态中。我们不需要管它。

    2、Exception是程序本身能够处理的异常 Exception类是所有异常类的父类,其子类对应了各种各样可能出现的异常事件。 通常Java的异常可分为:

          1. RuntimeException 运行时异常  运行期间抛出 比如空指针

    这类异常通常是由编程错误导致的,所以在编写程序时,并不要求必须使用异常处理机制来处理这类异常,经常需要通过增加“逻辑处理来避免这些异常”。如被 0 除、数组下标越界、空指针、强转错误等,其产生比较频繁,处理麻烦,如果显式的声明或捕获将会对程序可读性和运行效率影响很大。 因此由系统自动检测并将它们交给缺省的异常处理程序(用户可不必对其处理)。

          2. CheckedException 已检查异常  编译期间处理 比如io

    所有不是RuntimeException的异常,统称为Checked Exception,又被称为“已检查异常”,如IOException、SQLException等以及用户自定义的Exception异常。 这类异常在编译时就必须做出处理,否则无法通过编译。

    2、异常的处理方式之一:捕获异常

     try-catch执行流程

    多重 catch 语句中,异常类型必须子类在前父类在后

    多重 catch 语句中,异常类型必须子类在前父类在后,如果你把父类放前面就执行不到后边的了,

    把子类放前面,因为子类的异常信息比父类更详细

        public static void readMyFile() {

            FileReader reader = null;

            try {

                reader = new FileReader("d:/b.txt"); // 没有b.txt这个文件的异常

                System.out.println("step1");

                char c1 = (char) reader.read();    // 文件中读取不到内容异常

                System.out.println(c1);

     

            } catch (FileNotFoundException e) {   // 子类异常在父类异常前面

                System.out.println("step2");

                e.printStackTrace();

            } catch (IOException e) {

                e.printStackTrace();

            } finally {

                System.out.println("step3");

                try {                             // 如果 没有b.txt文件 reader为null 可能出现空指针异常

                    if (reader != null) {

                       reader.close();

                    }

                } catch (IOException e) {

                    e.printStackTrace();

                }

            }

        }

     

    }

    3、 异常的处理方式之二:声明异常(throws子句)

    抛到最上层 谁调谁执行

    public class Test03 {

        public static void main(String[] args) throws IOException {

           readMyFile();

        }

     

        public static void readMyFile() throws IOException {  //此处这下面异常的最顶层异常

           FileReader reader = null;

           reader = new FileReader("d:/b.txt");  //FileNotFoundException异常

           System.out.println("step1");

           char c1 = (char) reader.read();       //IOException异常

           System.out.println(c1);

     

           if (reader != null) {

               reader.close();                   //IOException异常

           }

        }

    }

    4、自定义异常

    package exception01;

     

    public class Test04 {

     

        public static void main(String[] args) throws IllegalAgeException  {

           Person p = new Person();

           p.setAge(-10);

        }

    }

     

    class Person {

        private int age;

     

        public void setAge(int age) throws IllegalAgeException  {

           if (age < 0) {

               throw new IllegalAgeException("年龄不能为负数");

           }

           this.age = age;

        }

     

    }

    // 继承运行时异常 上面不用处理try-catch or throws

    // 继承Exception->不是运行时异常->必须在编译的时候就处理

    class IllegalAgeException extends Exception {

        public IllegalAgeException() {           

        }

     

        public IllegalAgeException(String message) {

           super(message);

        }

    }

    5、补充说明

    1.JDBC关闭连接的时候为什么要单独try-catch

    把所有关闭语句写在同一个try块里面,一旦前面的关闭语句抛异常,后面的关闭语句就无法执行了,所以不能这样写,要给每个关闭语句一个try块。

        //关闭资源

        public static void closeAll(ResultSet rs,Statement stmt,Connection conn){

           try {

               rs.close();

           } catch (Exception e) {

              

           }

           try {

               stmt.close();

           } catch (SQLException e) {

              

           }

           try {

               conn.close();

           } catch (SQLException e) {

              

           }

        }

    总结:https://www.cnblogs.com/wangyingli/p/5912269.html

    1.若一段代码前有异常抛出,并且这个异常没有被捕获,这段代码将产生编译时错误「无法访问的语句」。如代码1

    2.若一段代码前有异常抛出,并且这个异常被try...catch所捕获,若此时catch语句中没有抛出新的异常,则这段代码能够被执行,否则,同第1条。如代码2

    3.若在一个条件语句中抛出异常,则程序能被编译,但后面的语句不会被执行。如代码3

    问题总结:

    问题1:为什么在UserServiceImpl中要显示的创建无参构造方法?

    答:因为父类构造器抛出了异常 子类在初始化的时候 子类的构造器会调用父类的构造器(初始化的时候总是先初始化父类) 父类的构造器抛出了异常,子类的构造器在调用父类的构造器时也应该抛出异常(并且该异常可以比父类异常范围大),若子类不提供构造器,虽然会隐式的生成无参构造器,但是生成的无参构造器不会抛出父类构造器的异常,所以子类必须显示的声明无参构造器来抛出此异常。

    子类

    public class UserServiceImpl extends UnicastRemoteObject  implements UserService{

     

        protected UserServiceImpl() throws RemoteException {

             super();

        }

    }

    父类:

        protected UnicastRemoteObject() throws RemoteException

        {

            this(0);

        }

    问题2:若一个类只有有参构造器,没有无参构造器,能用反射创建对象吗?

                   答:不能,因为反射在创建对象时,必须要走无参构造器。如果是用new关键字,则可以直接调用有参构造器创建对象。

    问题3:接口为什么也要抛出异常

                   答:当实现类在实现接口方法时可能会抛出异常,根据java语法规范,该接口也需要手动抛出对应的异常。

  • 相关阅读:
    第十三周总结
    第十二周总结
    团队冲刺第十五天-KeepRunningAPP
    团队第一阶段冲刺评价
    团队项目KeepRunning第一阶段成果展示
    团队冲刺第十四天-KeepRunningAPP
    第十一周总结
    团队冲刺第十三天-KeepRunningAPP
    CVPR2018关键字生成词云图
    Delphi APP 開發入門(八)SQLite資料庫
  • 原文地址:https://www.cnblogs.com/lanmao123/p/10486692.html
Copyright © 2020-2023  润新知