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给我们提供的异常类:
下面我来说一下异常的分类:
- 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);
}
}
运行结果:
总结
写到这,这篇异常处理的文章就写完了,看完后我们应该明白以下几个问题:
- 什么是异常
- 出现异常如何处理
- 如何自定义异常
可能写的有点啰嗦了,但对新手是友好的,还请大家多多包涵,希望能对大家有所帮助。