Java匹马行天下之JavaSE核心技术——异常处理

Java匹马行天下之JavaSE核心技术——异常处理

异常的简介

在Java中,异常就是Java在编译、运行或运行过程中出现的错误。

程序错误分为三种:编译错误运行时错误逻辑错误

  • 编译错误是因为程序没有遵循语法规则,编译程序能够自己发现并且提示我们错误的原因和位置,这个也是新手在刚接触编程语言时经常遇到的问题。
  • 运行时错误是因为程序在执行时,运行环境发现了不能执行的操作。
  • 逻辑错误是因为程序没有按照预期的逻辑顺序执行。异常也就是指程序运行时发生错误,而异常处理就是对这些错误进行处理和控制。

下面看一段代码

package com.yichang;

public class Test {
    public static void main(String[] args) {
        System.out.println(1/0);//0不能做除数
    }
}

运行后的结果为

 

我们发现程序出了错,而图中的错误信息告诉我们两个信息:

  • 出了什么错
  • 出错的位置

异常产生的过程

以上面的代码为例,程序在运行过程中,先运行main方法,然后执行到1/0时,程序就会报错,程序先会创建一个错误对象,然后把这个错误对象丢出来,我们都知道我们的程序是运行在Java虚拟机(JVM)上,程序丢出来的错误对象就会被JVM捕获到。当然,JVM捕获到错误对象后,它也不知道该怎么办,它不可能帮你调错,JVM会把错误信息给你打印出来。

总结一下,就是:

  • 上例中出现的异常是运行时异常(异常是错误)
  • 程序会创建一个错误对象,然后把错误对象丢出来(抛异常)
  • 默认由JVM把错误信息进行捕获,打印出来(捕获异常)

为什么不能直接让JVM获取异常

先看下面代码

package com.yichang;

public class Test {
    public static void main(String[] args) {
        System.out.println(1/0);//0不能做除数

        System.out.println("你好");
    }
}

运行结果为

 

 

从上面代码和运行结果中,我们不难看出在控制台没有打印出你好。出现这种情况,是因为在JVM捕获到异常后,程序会终止。换句话说,在由JVM来处理错误的时候,此时,程序会终止,因此异常之后的代码就无法运行了。我们可以根据生活实例想想,什么时候百度会因为搜不到东西停服务,肯定是不会的,因此我们后面要做的就是如何在异常到达JVM之前把异常拦下来,自己单独处理,就不要麻烦JVM了。

异常的分类

在上面实例中我说过程序会创建错误对象,说到对象,我们都知道对象是由创建的,那异常对象肯定是通过异常类来创建的。下面图中就是Java给我们提供的异常类:

 

 

 
从图中我们能看出Throwable是所有异常的根,所有的异常类都继承自Throwable,就像面向对象里面所有的类都继承自Object

 

下面我来说一下异常的分类

  • RuntimeException:运行时异常,一般不手动处理,出问题了再处理。
  • 其他Exception:必须要经过手动处理。
  • Error:一般指的是系统级错误。

我再用生活例子解释一下这三种异常,便于新手理解: 假设我们现在开车上山

  • 开车的过程中发现山上有许多小石头,但我们不可能把所有小石头都处理了,这时候我们依旧正常开,什么时候小石头把车胎给弄坏了,我们再下来,换备胎,这就是运行时异常
  • 开车的过程中发现前面有一个很大的石头挡住了路,这时候你必须下车先把这石头挪走,你才能继续上山,这也就是其他Exception
  • 开车上山的过程中山塌陷了,你又无法处理,必须要等到山好了你才能继续出发,也就是Error,就是这种错误我们一般的程序员是处理不了的。

异常的处理方法

try···catch处理

语法

try{
    //尝试执行的代码
        }catch(Exception e){

    //处理异常的代码
        }finally{

    //最终的
        }

下面我将文中案例进行改造

package com.yichang;

public class Test {
    public static void main(String[] args) {
        try {
            System.out.println(1/0);//0不能做除数
        }catch (Exception e){
            e.printStackTrace();//打印错误信息,给程序员看的
            System.out.println("系统出现错误,请联系管理员");//给客户看的
        }finally {

            //一般做收尾工作
            System.out.println("你好");
        }

    }
}

运行结果为

 

 

  • try— 用于监听。将要被监听的代码(可能抛出异常的代码)放在try语句块之内,当try语句块内发生异常时,异常就被抛出。
  • catch— 用于捕获异常。catch用来捕获try语句块中发生的异常。
  • finally— finally语句块总是会被执行。它主要用于回收在try块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。

throws和throw处理

  • throws表示方法准备要扔出来一个异常
  • throw表示向外抛出异常

举例说明

package com.yichang;

public class Test1 {

    public static void chu(int a,int b) throws Exception{
        if (b==0){
            throw new Exception("除数不能为0");
        }else {
            System.out.println(a/b);
        }
    }

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

        chu(1,0);
    }
}

  

运行结果

 

 

以上这两种方法都是处理异常的,如果这个异常你可以处理,就用try···catch方法捕获并处理异常,如果这个异常你不能处理,就用throws方法抛出异常,但作为程序员的我们要始终记住一句话:产生的错误尽可能的自己处理,少向外抛出异常

自定义异常

到这可能有的朋友要问了,为什么要自定义异常,Java给的那么多还不够用吗?我可以告诉你,当然不够用,比如在生活中,我们都知道外面的澡堂子里边是分男女澡堂的,如果有男顾客走进了女澡堂或有女顾客走进了男澡堂,就坏事了,这算是一个大异常吧,那大家想,jdk会给我们提供跟性别还有澡堂子有关的异常吗?肯定是不可能的,那此时就需要我们自定义异常。我以澡堂子为例,写一个程序,供大家参考。

自定义异常:直接继承Exception或者RuntimeException来是实现自定义异常

Person类

package com.yichang;

public class Person {
    String name;//姓名
    String gender;//性别

    public Person(String name, String gender) {
        this.name = name;
        this.gender = gender;
    }
}

ZaoTangZi类

package com.yichang;

public class ZaoTangZi {
    public void man(Person p) throws GenderException{
        if (p.gender=="男"){
            System.out.println("欢迎光临");
        }else {
            //需要抛出一个异常
            throw new GenderException("性别错误,这里是男澡堂子");
        }
    }
}

GenderException类

package com.yichang;

public class GenderException extends Exception{
    //自己定义的异常必须要继承Exception或RuntimeException

    public GenderException(String msg){
        super(msg);//调用父类的构造方法,Exception(msg)
    }
}

Test类

package com.yichang;

public class Test {
    public static void main(String[] args) throws GenderException{
        Person p1 = new Person("张三","男");
        Person p2 = new Person("小花","女");

        ZaoTangZi z = new ZaoTangZi();
        z.man(p2);
    }
}

运行结果:

 

 

总结

写到这,这篇异常处理的文章就写完了,看完后我们应该明白以下几个问题:

  • 什么是异常
  • 出现异常如何处理
  • 如何自定义异常

可能写的有点啰嗦了,但对新手是友好的,还请大家多多包涵,希望能对大家有所帮助。