JavaSE----异常及异常处理

Java 异常概念

    前言: 在使用计算机语言进行项目开发的过程中,即使程序员把代码写得尽善 尽美,在系统的运行过程中仍然会遇到一些问题,因为很多问题不是靠 代码能够避免的,比如:客户输入数据的格式,读取文件是否存在,网络是否始终保持通畅等等的诸多问题。
    异常: 在Java语言中,将程序执行中发生的不正常情况称为“异常”。Java程序在执行过程中所发生的异常(运行时一切不正常情况)事件可分为两类:广义上的异常(Error)和狭义上的异常(Exception
    Error: Java虚拟机无法解决的严重问题。如:JVM系统内部错误、资源耗尽等严重情况。一般不编写针对性的代码进行处理(基本上都是外界因素的影响)。
    Exception: 其它因编程错误或偶然的外在因素导致的一般性问题,可 以使用针对性的代码进行处理
比如说:

public class ExcptionTest { 
  public void test(){ 
     System.out.println( "====运行前====" ); 
     int a=10/0; 
     System.out.println( "====运行后====" ); 
   }
 public static void main(String[] args) { 
     ExcptionTest et= new ExcptionTest(); 
     et.test(); 
   } 
}

异常的结果:在这里插入图片描述

异常的体系

    异常分为运行期异常编译期异常两种

    运行期异常:程序运行时抛除的异常,所有RuntimeException的子类都 是运行期异常,比如数学异常 、空指针异常 、 数组下标越界、等等…
    编译器异常:编译期异常(Checked Exception):除去运行期的异常都是编译期异常, 也称为检测异常(也就是编译器在你敲完一段代码后打红线的部分

    Exception异常类、Error类都是直接继承Throwable父类
在这里插入图片描述
    我们现在常见的异常就是有:ArrayIndexOutOfBoundsException 数组索引异常StringIndexOutOfBoundsException 字符串索引异常NumberFormatException 数字化格式异常NullPointerException 空指针异常ClassCastException 类的类型转换异常ArithmeticException、除数为0异常

public static void main(String[] args) {
        /**
         * ArrayIndexOutOfBoundsException 索引异常
         * 抛出以表示使用非法索引访问数组。 索引为负数或大于或等于数组的大小。
         * Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 5
         * at Demo1.Demo1D1.main(Demo1D1.java:15)
         */
        //int []arr1 = new int[]{1,2,3,4,5};
        //System.out.println(arr1[5]);

        /**
         * StringIndexOutOfBoundsException  索引异常
         * 抛出String方法来指示索引为负或大于字符串的大小。 对于某些方法(如charAt方法),当索引等于字符串的大小时,也会抛出此异常。
         * Exception in thread "main" java.lang.StringIndexOutOfBoundsException: String index out of range: 3
         * 	at java.lang.String.charAt(String.java:658)
         * 	at Demo1.Demo1D1.main(Demo1D1.java:23)
         */
        //String str1 = "123";
        //System.out.println(str1.charAt(3));

        /**
         * NumberFormatException数字化格式异常
         * 抛出以表示应用程序已尝试将字符串转换为其中一个数字类型,但该字符串不具有相应的格式。
         *Exception in thread "main" java.lang.NumberFormatException: For input string: "a"
         * 	at java.lang.NumberFormatException.forInputString(NumberFormatException.java:65)
         * 	at java.lang.Integer.parseInt(Integer.java:580)
         * 	at java.lang.Integer.<init>(Integer.java:867)
         * 	at Demo1.Demo1D1.main(Demo1D1.java:36)
         */
        //String str = "a";
        //Integer integer = new Integer(str);

        /**
         * NullPointerException空指针异常
         * 当应用程序尝试在需要对象的情况下使用null时抛出。 这些包括:
         * 调用一个null对象的实例方法。
         * 访问或修改null对象的字段。
         * 取null的长度,好像是一个数组。
         * 访问或修改的时隙null就好像它是一个数组。
         * 投掷null好像是一个Throwable价值。
         *
         *Exception in thread "main" java.lang.NullPointerException
         * 	at Demo1.Demo1D1.main(Demo1D1.java:48)
         */
        //String str = null;
        //System.out.println(str.length());

        /**
         *ClassCastException 类的类型转换异常
         * 抛出表示代码尝试将对象转换为不属于实例的子类
         * Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String
         * 	at Demo1.Demo1D1.main(Demo1D1.java:63)
         */
        //Object obj = new Integer(0);
        //System.out.println((String)obj);

        /**
         * 广义异常:
         */
        //System.out.println(sum(5));
}

异常处理

    Java编程语言使用异常处理机制为程序提供了错误处理的能力。Java的异常处理是通过5个关键字来实现的:try、catch、 finally、 throw、throws
在这里插入图片描述

public static void main(String[] args) {
        try{
        /*
        检测不安全的代码块(发现异常) try块中任何一条语句发生了异常,
        下面的代码将不会被执行,程序将 跳转到异常处理代码块中,
        即catch块。因此,不要随意将不相关的代 码放到try块中,
        因为随时可能会中断执行。
        */
            Scanner scanner = new Scanner(System.in);
            System.out.println("输入两个数字");
            int a = scanner.nextInt();
            int b = scanner.nextInt();
            System.out.println(a/b);
        }catch (ArithmeticException e){
        /*
        把抓到的类型匹配的异常捕获,
        保证程序能继续运行下去 catch语句必须紧跟着try语句之后,
        称为捕获异常,也就是异常处理函数, 
        一个try后面可以写多个catch,
        分别捕获不同类型的异常,要从子类往父类 的顺序写,否则有编译错误
        */
            e.printStackTrace();
            /**
             * 拿出异常
             * 
             */
            System.out.println("0不能作为除数");
        }
    }

       getMessage() 获取异常信息,返回字符串
       printStackTrace() 获取异常类名和异常信息,以及异常出现在程序中的 位置。返回值void
       Finally关键字:
      与final很像,但是完全不是一个意思。
final是在面向对象语法时出现的关键字,其用于修饰属性、方法和类

  • 被final修饰的成员对象必须在定义时赋初值,或者用构造方法为其赋值,其值后期不能改变
  • 被final修饰的静态变量必须在定义的时候就就赋初值,且不能改变
  • 被final修饰的成员方法不能被重写
  • 被final修饰的类不能被继承

finally是在异常处理时会用到的关键字。

  • finally语句往往写在try后、catch后。
  • 先执行try语句或者catch的异常
  • 但是执行末尾时总会去执行finally语句
  • finally语句不管在什么条件下都会执行完尽管执行后不再执行try或者catch里面剩余的语句

请欣赏下面一段代码(可以直接放在编译器上运行)

/**
 * finally的基本语法
 * try{
 *     可能会发生异常的代码
 *   }catch(异常类型 引用名){
 *   异常处理代码
 *   }finally{
 *   必须执行代码
 *   }
 *
 *   finally该内ly容总是会执行的,只能有一个finally语句
 */
public class DemoFinally {
    public static void main(String[] args) {
        try{
            int a = 10;
            int b = 0;
            System.out.println(a/b);
            System.out.println("aaaaa");
        }catch(ArithmeticException e){
            e.printStackTrace();
            System.out.println("出错啦");
        }finally{//不管程序有没有异常,都会执行finally的语句
            System.out.println("bbbbbb");
        }
        //System.out.println(m1(10,0));
        /**
         * 如果try体系中没有catch语句来捕获异常,但是有finally语句
         * 如果出现异常,那么程序会在报错前执行完finally的语句,最后停止
         */
        /*try{
            int m = 5;
            int n = 0;
            System.out.println(m/n);
        }finally {
            System.out.println("aaaaaa");
        }
        System.out.println("aaaa");*/
    }
    public static int m1(int a,int b){
        try{
            int c=a/b;
            return c;
        }catch (ArithmeticException e){
            e.printStackTrace();
            System.out.println("出错啦");
            return -1;
        }finally{//在return之前,总会执行finally语句
            System.out.println("finally");
            //既是finally也是一段return,程序也是会执行,然后结束;
            return 0;
        }
    }
}

throws 和 throw

       Throws:定义一个方法的时候可以使用throws关键字声明 (必须放在方法名的括号后) 表示此方法 不处理异常,而交给方法调用处进行处理。 例如: public void test throws 异常1,异常2,异常3{ }。但是这种用法一般只会出现在基层的方法中,而顶层方法就不能再抛,在抛就抛给虚拟机,虚拟机哪会处理异常,他就直接给你报错。而且任何方法都可以使用throws关键字声明异常类型,包括抽象方法。
       子类重写父类中的方法,子类方法不能声明抛出比父类类型更大的异常(针对编 译期异常)。 使用了throws的方法,调用时必须处理声明的异常,要么使用try-catch,要 么继续使用throws声明。顶层只能用try-catch
       Throw:throw关键字用于显式抛出异常,抛出的时候是抛出的是一个异常类的实 例化对象. 在异常处理中,try语句要捕获的是一个异常对象,那么此异常对象也 可以自己抛出。throw往往写在语句中,而非方法名后面。自定义异常就是自己定义的异常类,也就是API中的标准异常类的直接或间接的 子类

 public static void main(String[] args) {
         try{
             m1();
         }catch (NullPointerException e){
             e.printStackTrace();
         }
         System.out.println("aaa");

    }
    public static void m1(){
        System.out.println(m2(null));

    }
    public static int m2(String a){
        if(a==null){//如果报错,则直接抛出一个异常类的对象
            throw new NullPointerException("空指针异常");
        }
        return a.length();
    }

异常代码:

  Exception in thread "main" java.lang.NullPointerException: 空指针异常
  at Demo1.异常处理.DemoThrow.m2(DemoThrow.java:17)
  at Demo1.异常处理.DemoThrow.m1(DemoThrow.java:12)
  at Demo1.异常处理.DemoThrow.main(DemoThrow.java:9)

       Throw和Throws的区别:

  • throw用于 方法体中,用来抛出一个实际的异常对象,使用throw后, 要么使用try catch捕获异常,要么使用throws声明异常

  • throws用于 方法声明处,用来声明该方法可能发生的异常类型,可以是 多个异常类型,用来强制调用该方法时处理这些异常

  • Throw写在方法体时,明确的告诉你这会抛出异常

  • 它抛出的异常基本上由自己内部的方法快来处理掉,有Throw必有Throws

  • Throws写在方法名括号的后面用来显示异常

  • Throws会表示抛出异常的类型,往往由上一级(调用次方法的)来try catch解决异常而自身不解决

自定义异常

    自定义异常就是自己定义的异常类,也就是API中的标准异常类的直接或间接的子类。自定义异常类中往往不写其他方法,只重载需要使用的构造方法。继承Exception,在方法中使用throw抛出后,必须在方法中try-catch或 throws抛出

public class DemoCreatThrow extends Exception{
    public DemoCreatThrow(){
        super();
    }
    public DemoCreatThrow(String a){
        super(a);
    }
}

    作用:用自定义异常标记业务逻辑的异常,避免与标准异常混淆
代码演示:

class DemoThrow1{
    public static void main(String[] args) {
        try{
            System.out.println(m1(10,0));
        } catch (DemoCreatThrow e) {
            e.printStackTrace();
        }
    }
    public static int m1(int a,int b) throws DemoCreatThrow {
        if(b==0){
            throw new DemoCreatThrow("自定义的异常");
        }
        return a/b;
    }
}

在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值