zoukankan      html  css  js  c++  java
  • java之异常处理、异常分类、Throwable、自定义异常

    参考http://how2j.cn/k/exception/exception-trycatch/336.html

    异常处理

    try catch

    1.将可能抛出FileNotFoundException 文件不存在异常的代码放在try里
    2.如果文件存在,就会顺序往下执行,并且不执行catch块中的代码
    3. 如果文件不存在,try 里的代码会立即终止,程序流程会运行到对应的catch块中
    4. e.printStackTrace(); 会打印出方法的调用痕迹,如此例,会打印出异常开始于TestException的第16行,这样就便于定位和分析到底哪里出了异常

    package exception;
     
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
     
    public class TestException {
     
        public static void main(String[] args) {
             
            File f= new File("d:/LOL.exe");
             
            try{
                System.out.println("试图打开 d:/LOL.exe");
                new FileInputStream(f);
                System.out.println("成功打开");
            }
            catch(FileNotFoundException e){
                System.out.println("d:/LOL.exe不存在");
                e.printStackTrace();
            }
             
        }
    }

    使用异常的父类进行catch

    FileNotFoundException是Exception的子类,使用Exception也可以catch住FileNotFoundException

    package exception;
      
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
      
    public class TestException {
      
        public static void main(String[] args) {
              
            File f= new File("d:/LOL.exe");
              
            try{
                System.out.println("试图打开 d:/LOL.exe");
                new FileInputStream(f);
                System.out.println("成功打开");
            }
             
            catch(Exception e){
                System.out.println("d:/LOL.exe不存在");
                e.printStackTrace();
            }
              
        }
    }

    多异常捕捉办法1

    有的时候一段代码会抛出多种异常,比如

    new FileInputStream(f);
    Date d = sdf.parse("2016-06-03");

    这段代码,会抛出 文件不存在异常 FileNotFoundException 和 解析异常ParseException
    解决办法之一是分别进行catch

    catch (FileNotFoundException e) {
        System.out.println("d:/LOL.exe不存在");
        e.printStackTrace();
    } catch (ParseException e) {
        System.out.println("日期格式解析错误");
        e.printStackTrace();
    }
    package exception;
     
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.text.ParseException;
    import java.text.SimpleDateFormat;
    import java.util.Date;
     
    public class TestException {
     
        public static void main(String[] args) {
     
            File f = new File("d:/LOL.exe");
     
            try {
                System.out.println("试图打开 d:/LOL.exe");
                new FileInputStream(f);
                System.out.println("成功打开");
                SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
                Date d = sdf.parse("2016-06-03");
            } catch (FileNotFoundException e) {
                System.out.println("d:/LOL.exe不存在");
                e.printStackTrace();
            } catch (ParseException e) {
                System.out.println("日期格式解析错误");
                e.printStackTrace();
            }
        }
    }

    多异常捕捉办法2

    另一个种办法是把多个异常,放在一个catch里统一捕捉

    catch (FileNotFoundException | ParseException e) {

    这种方式从 JDK7开始支持,好处是捕捉的代码更紧凑,不足之处是,一旦发生异常,不能确定到底是哪种异常,需要通过instanceof 进行判断具体的异常类型

    if (e instanceof FileNotFoundException)
        System.out.println("d:/LOL.exe不存在");
    if (e instanceof ParseException)
        System.out.println("日期格式解析错误");
    package exception;
     
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.text.ParseException;
    import java.text.SimpleDateFormat;
    import java.util.Date;
     
    public class TestException {
     
        public static void main(String[] args) {
     
            File f = new File("d:/LOL.exe");
     
            try {
                System.out.println("试图打开 d:/LOL.exe");
                new FileInputStream(f);
                System.out.println("成功打开");
                SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
                Date d = sdf.parse("2016-06-03");
            } catch (FileNotFoundException | ParseException e) {
                if (e instanceof FileNotFoundException)
                    System.out.println("d:/LOL.exe不存在");
                if (e instanceof ParseException)
                    System.out.println("日期格式解析错误");
     
                e.printStackTrace();
            }
     
        }
    }

    finally

    无论是否出现异常,finally中的代码都会被执行

    throws

    考虑如下情况:
    主方法调用method1
    method1调用method2
    method2中打开文件

    method2中需要进行异常处理
    但是method2不打算处理,而是把这个异常通过throws抛出去
    那么method1就会接到该异常。 处理办法也是两种,要么是try catch处理掉,要么也是抛出去
    method1选择本地try catch住 一旦try catch住了,就相当于把这个异常消化掉了,主方法在调用method1的时候,就不需要进行异常处理了

    package exception;
     
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
     
    public class TestException {
     
        public static void main(String[] args) {
            method1();
     
        }
     
        private static void method1() {
            try {
                method2();
            } catch (FileNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
     
        }
     
        private static void method2() throws FileNotFoundException {
     
            File f = new File("d:/LOL.exe");
     
            System.out.println("试图打开 d:/LOL.exe");
            new FileInputStream(f);
            System.out.println("成功打开");
     
        }
    }

    throw和throws的区别

    throws与throw这两个关键字接近,不过意义不一样,有如下区别:
    1. throws 出现在方法声明上,而throw通常都出现在方法体内。
    2. throws 表示出现异常的一种可能性,并不一定会发生这些异常;throw则是抛出了异常,执行throw则一定抛出了某个异常对象。

    练习-异常处理

    假设有一个方法 public int method(), 会返回一个整数
    在这个方法中有try catch 和 finally.
    try 里返回 1
    catch 里 返回 2
    finally 里 返回3
    那么,这个方法到底返回多少?

    public static void main(String[] args) {
            System.out.println(test());
        }
    
        public static int test() {
            try {
                return 1;
            }catch (Exception e){
                return 2;
            }finally {
                return 3;
            }
        }

    方法返回值是:3

    原因

    遇到try时,运行代码,执行里面的代码,包括return的代码;

    由于finally肯定执行,执行finally代码,返回return 3 结束。

    异常分类

    异常分类: 可查异常,运行时异常和错误3种
    其中,运行时异常和错误又叫非可查异常

    可查异常

    可查异常: CheckedException
    可查异常即必须进行处理的异常,要么try catch住,要么往外抛,谁调用,谁处理,比如 FileNotFoundException
    如果不处理,编译器,就不让你通过

    package exception;
      
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
      
    public class TestException {
      
        public static void main(String[] args) {
              
            File f= new File("d:/LOL.exe");
              
            try{
                System.out.println("试图打开 d:/LOL.exe");
                new FileInputStream(f);
                System.out.println("成功打开");
            }
            catch(FileNotFoundException e){
                System.out.println("d:/LOL.exe不存在");
                e.printStackTrace();
            }
              
        }
    }

    运行时异常

    运行时异常RuntimeException指: 不是必须进行try catch的异常
    常见运行时异常:
    除数不能为0异常:ArithmeticException
    下标越界异常:ArrayIndexOutOfBoundsException
    空指针异常:NullPointerException
    在编写代码的时候,依然可以使用try catch throws进行处理,与可查异常不同之处在于,即便不进行try catch,也不会有编译错误
    Java之所以会设计运行时异常的原因之一,是因为下标越界,空指针这些运行时异常太过于普遍,如果都需要进行捕捉,代码的可读性就会变得很糟糕。

    package exception;
      
    public class TestException {
      
        public static void main(String[] args) {
             
            //任何除数不能为0:ArithmeticException
            int k = 5/0;
             
            //下标越界异常:ArrayIndexOutOfBoundsException
            int j[] = new int[5];
            j[10] = 10;
             
            //空指针异常:NullPointerException
            String str = null;
            str.length();
       }
    }

    错误

    错误Error,指的是系统级别的异常,通常是内存用光了
    默认设置下,一般java程序启动的时候,最大可以使用16m的内存
    如例不停的给StringBuffer追加字符,很快就把内存使用光了。抛出OutOfMemoryError
    与运行时异常一样,错误也是不要求强制捕捉的

    package exception;
      
    public class TestException {
      
        public static void main(String[] args) {
         
            StringBuffer sb =new StringBuffer();
             
            for (int i = 0; i < Integer.MAX_VALUE; i++) {
                sb.append('a');
            }
             
        }
     
    }

    三种分类

    总体上异常分三类:
    1. 错误
    2. 运行时异常
    3. 可查异常

    练习-异常分类

    运行时异常 RuntimeException,能否被捕捉?
    错误Error,能否被捕捉?
    面试题常问题:运行时异常与非运行时异常的区别

    面试题常问题:运行时异常与非运行时异常的区别:
    运行时异常是不可查异常,不需要进行显式的捕捉
    非运行时异常是可查异常,必须进行显式的捕捉,或者抛出
    不要答成:
    运行时异常是运行的时候抛出的异常,非运行时异常,不运行也能抛出

    package exception;
     
    public class TestException {
     
        public static void main(String[] args) {
     
            String str = null;
     
            try {
                str.toString();
            } catch (NullPointerException e) {
                System.out.println("捕捉到运行时异常: NullPointerException ");
            }
     
            StringBuffer sb = new StringBuffer("1234567890");
            try {
                for (int i = 0; i < 100; i++) {
                    sb.append(sb.toString());
                }
            } catch (OutOfMemoryError e) {
                System.out.println("捕捉到内存用光错误:  OutOfMemoryError");
            }
     
        }
    }

    Throwable

    Throwable

     Throwable是类,Exception和Error都继承了该类
    所以在捕捉的时候,也可以使用Throwable进行捕捉
    如图: 异常分Error和Exception
    Exception里又运行时异常可查异常

    package exception;
     
    import java.io.File;
    import java.io.FileInputStream;
     
    public class TestException {
     
        public static void main(String[] args) {
     
            File f = new File("d:/LOL.exe");
     
            try {
                new FileInputStream(f);
                //使用Throwable进行异常捕捉
            } catch (Throwable t) {
                // TODO Auto-generated catch block
                t.printStackTrace();
            }
     
        }
    }

    练习-Throwable

    在方法声明上,可以抛出指定的异常,比如FileNotFoundException
    那么能否抛出Throwable这个类?
    这个方法的调用者又该如何处理?

    答案:可以抛出

    public class TestException {
        public static void main(String[] args) {
            method1();
        }
         
        public static void method1() {
            try{
                method2();
            }
            catch(Throwable t) {
                System.out.println("系统打开失败");
                t.printStackTrace();
            }
        }
         
        public static void method2() throws Throwable{
            File f = new File("d:/LOL.exe");
             
            System.out.println("系统先尝试打开");
            new FileInputStream(f);
            System.out.println("系统打开成功");
        }
    }

    注意:将Throwable 换成Exception就不行了,原因是:比如,它万一抛出来一个error,你让人家怎么捕获?前面类型转换的内容,这里将子类转父类,是不一定行的,相反的父类转子类是可行的。

    自定义异常

    创建自定义异常

    一个英雄攻击另一个英雄的时候,如果发现另一个英雄已经挂了,就会抛出EnemyHeroIsDeadException
    创建一个类EnemyHeroIsDeadException,并继承Exception
    提供两个构造方法
    1. 无参的构造方法
    2. 带参的构造方法,并调用父类的对应的构造方法

    class EnemyHeroIsDeadException extends Exception{
         
        public EnemyHeroIsDeadException(){
             
        }
        public EnemyHeroIsDeadException(String msg){
            super(msg);
        }
    }

    抛出自定义异常

    在Hero的attack方法中,当发现敌方英雄的血量为0的时候,抛出该异常
    1. 创建一个EnemyHeroIsDeadException实例
    2. 通过throw 抛出该异常
    3. 当前方法通过 throws 抛出该异常

    在外部调用attack方法的时候,就需要进行捕捉,并且捕捉的时候,可以通过e.getMessage() 获取当时出错的具体原因

    package charactor;
      
    public class Hero {
        public String name;
        protected float hp;
     
        public void attackHero(Hero h) throws EnemyHeroIsDeadException{
            if(h.hp == 0){
                throw new EnemyHeroIsDeadException(h.name + " 已经挂了,不需要施放技能" );
            }
        }
     
        public String toString(){
            return name;
        }
         
        class EnemyHeroIsDeadException extends Exception{
             
            public EnemyHeroIsDeadException(){
                 
            }
            public EnemyHeroIsDeadException(String msg){
                super(msg);
            }
        }
          
        public static void main(String[] args) {
             
            Hero garen =  new Hero();
            garen.name = "盖伦";
            garen.hp = 616;
     
            Hero teemo =  new Hero();
            teemo.name = "提莫";
            teemo.hp = 0;
             
            try {
                garen.attackHero(teemo);
                 
            } catch (EnemyHeroIsDeadException e) {
                // TODO Auto-generated catch block
                System.out.println("异常的具体原因:"+e.getMessage());
                e.printStackTrace();
            }
             
        }
    }

    练习-自定义异常

  • 相关阅读:
    leetcode 第41题 Trapping Rain Water
    leetcode第40题--First Missing Positive
    leetcode第39题--Combination Sum II
    leetcode第38题--Combination Sum
    leetcode第37题--Count and Say
    17_7_18 servlet 动态工程 先加载servlet request注意事项
    17_7_13 Servlet分页思路
    17_7_12 tomcat 安装+(java+tomcat)环境变量+windows安装报错:JRE_HOME
    17_7_12 JDBC 中的 DBCP(链接池)和 Servlet Web工程之间的差异
    17_7_12 DBCP 中 ResultSet 判断是否有值 +判断 一个对象是否包含在数据库中
  • 原文地址:https://www.cnblogs.com/lijingran/p/9126125.html
Copyright © 2011-2022 走看看