[JAVA]异常
Posted HY_PIGIE
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[JAVA]异常相关的知识,希望对你有一定的参考价值。
目录
1.概念
在JAVA中,将程序执行过程中发生的不正常的行为称为‘异常’。
而关于异常,也有多种多样的。像是,算术异常,数组越界异常,空指针异常。
2.异常的体系结构
因为异常的种类繁多,为了对于不同的异常或者错误进行了很好的分类。
他们直接是父类与子类,继承的关系。
可以看到Thorwable是异常体系的顶端,派生出了Errow和Exception两个子类
Errow,指的是Java虚拟机无法解决的严重问题,比如:JVM的内部错误、资源耗尽等。
Exception,异常产生后程序员可以通过代码进行处理,使程序继续执行。
而在这两个子类下又还有许多的子类。
3.异常的分类
异常可能会在编译的时候发生错误,也可能是在运行时发生错误。根据这一情况,异常可以分为‘编译时异常’和‘运行时异常’。
3.1编译时异常
在程序编译期间发生的异常,称为编译时异常,也称为受检查异常(Checked Exception)
3.2运行时异常
在程序执行期间发生的异常,称为运行时异常,也称为非受检查异常(Unchecked Exception)
RunTimeException以及其子类对应的异常,都称为运行时异常。比如:NullPointerException、
ArrayIndexOutOfBoundsException、ArithmeticException
4.异常的处理
4.1防御式编程
LBYL:Look Before You Leap.(事前防御型)
即在每一次的操作之后都对其检查,查看是否发生异常。
EAFP:It's Easier to Ask Forgiveness than Permission.(事后认错型)
即是先把代码运行,当在运行过程种发生错误再捕获异常。
5.异常的抛出
在java中,可以通过throw关键字,抛出一个指定的异常对象,将错误信息更直白的告知
throw new 异常对象("异常的原因")
public class Test
public static void Printf(int[] arr)
if(arr == null)
throw new NullPointerException("传递的数组为空");
for (int a:arr
)
System.out.print(a + " ");
public static void main(String[] args)
int[] arr = 1,2,3,4,5,6,7;
Printf(null);
- throw必须放在方法的内部
- throw抛出的异常对象必须是Exception或者Exception的子类对象
- 如果抛出的是 RunTimeException 或者 RunTimeException 的子类,则可以不用处理,直接交给JVM来处理
- 异常一旦抛出,其后的代码就不会执行
6.异常的捕获
异常的捕获就是异常的助理方式:
1.异常声明throw
在方法的声明中使用throws关键字抛出,标明此方法有这个异常,但并没有处理此异常,让方法的调用者负责处理。在遇到异常后的代码不会运行
当方法的实现者没能力或不想解决的时候,可以使用throws抛出此异常给调用者解决。
public static void main(String[] args) throws IOException
当方法调用者遇到这种情况,可以选择处理或继续throws抛出异常
2.try-catch捕获处理
语法格式:
try
//可能会有异常的代码
//遇到异常后的代码不会执行
catch(要捕获的异常类型 e)
//如果try的类型与此处的catch匹配
//则会执行此catch下的代码
catch ()
//catch可以有多个
//如果catch成功捕获try处的异常,try-catch体系后的代码可以继续正常执行
如果try中出现了异常,且被catch捕获了,则try-catch的后续代码可以继续运行。如果try中出现了异常且没有被catch捕获,则编译器会报错,且后续的代码不会运行。
如果要捕获的异常之间存在子类与父类的关系,那么子类异常的catch一定要写在父类catch的前面。不然会捕获不到,子类的异常。
7.finally
在写程序时,有些特定的代码,不论程序是否发生异常,都需要执行,比如程序中打开的资源:网络连接、数据库连接、IO流等,在程序正常或者异常退出时,必须要对资源进进行回收。另外,因为异常会引发程序的跳转,可能导致有些语句执行不到,finally就是用来解决这个问题的。
语法格式:
try
// 可能会发生异常的代码
catch(异常类型 e)
// 对捕获到的异常进行处理
finally
// 此处的语句无论是否发生异常,都会被执行到
//后续的代码看try处的异常是否被catch捕获来决定是否执行
8.异常处理的流程
9.自定义异常类
示例:
继承RuntimeException表示为运行时异常
如果继承了Exception,默认为受查异常,则要在编译时进行处理,可以在方法中通过关键字throws抛出
import java.security.PrivateKey;
import java.util.Scanner;
class UserNameException extends RuntimeException//自定义一个类,继承RuntimeException
public UserNameException(String message) //带参数的构造方法,
super(message);
class PasswordException extends RuntimeException//自定义一个类,继承RuntimeException
public PasswordException(String message)
super(message);
public class Test
public static void Login()
String userName = "helloworld";
String userPassword = "123456789";
Scanner scanner = new Scanner(System.in);
System.out.println("请输入用户名:");
String name = scanner.nextLine();
System.out.println("请输入密码:");
String password = scanner.nextLine();
if(!name.equals(userName))
throw new UserNameException("用户名错误");
if(!password.equals(userPassword))
throw new PasswordException("密码错误");
System.out.println("登录成功");
public static void main(String[] args)
try
Login();
catch (UserNameException e)
e.printStackTrace();//printStackTrace()方法的意思是:在命令行打印异常信息在程序中出错的位置及原因
catch (PasswordException e)
e.printStackTrace();
printStackTrace()方法的意思是:在命令行打印异常信息在程序中出错的位置及原因
Java异常 Java异常简介及其架构
Java异常简介
Java异常是Java提供的一种识别及响应错误的一致性机制。
Java异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程序健壮性。在有效使用异常的情况下,异常能清晰的回答what, where, why这3个问题:异常类型回答了“什么”被抛出,异常堆栈跟踪回答了“在哪“抛出,异常信息回答了“为什么“会抛出。
Java异常机制用到的几个关键字:try、catch、finally、throw、throws。
• try -- 用于监听。将要被监听的代码(可能抛出异常的代码)放在try语句块之内,当try语句块内发生异常时,异常就被抛出。
• catch -- 用于捕获异常。catch用来捕获try语句块中发生的异常。
• finally -- finally语句块总是会被执行。它主要用于回收在try块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。只有finally块,执行完成之后,才会回来执行try或者catch块中的return或者throw语句,如果finally中使用了return或者throw等终止方法的语句,则就不会跳回执行,直接停止。
• throw -- 用于抛出异常。
• throws -- 用在方法签名中,用于声明该方法可能抛出的异常。
下面通过几个示例对这几个关键字进行简单了解。
示例一: 了解try和catch基本用法
public class Demo1 {
public static void main(String[] args) {
try {
int i = 10/0;
System.out.println("i="+i);
} catch (ArithmeticException e) {
System.out.println("Caught Exception");
System.out.println("e.getMessage(): " + e.getMessage());
System.out.println("e.toString(): " + e.toString());
System.out.println("e.printStackTrace():");
e.printStackTrace();
}
}
}
运行结果:
Caught Exception
e.getMessage(): / by zero
e.toString(): java.lang.ArithmeticException: / by zero
e.printStackTrace():
java.lang.ArithmeticException: / by zero
at Demo1.main(Demo1.java:6)
结果说明:在try语句块中有除数为0的操作,该操作会抛出java.lang.ArithmeticException异常。通过catch,对该异常进行捕获。
观察结果我们发现,并没有执行System.out.println("i="+i)。这说明try语句块发生异常之后,try语句块中的剩余内容就不会再被执行了。
示例二: 了解finally的基本用法
在"示例一"的基础上,我们添加finally语句。
public class Demo2 {
public static void main(String[] args) {
try {
int i = 10/0;
System.out.println("i="+i);
} catch (ArithmeticException e) {
System.out.println("Caught Exception");
System.out.println("e.getMessage(): " + e.getMessage());
System.out.println("e.toString(): " + e.toString());
System.out.println("e.printStackTrace():");
e.printStackTrace();
} finally {
System.out.println("run finally");
}
}
}
运行结果:
Caught Exception
e.getMessage(): / by zero
e.toString(): java.lang.ArithmeticException: / by zero
e.printStackTrace():
java.lang.ArithmeticException: / by zero
at Demo2.main(Demo2.java:6)
run finally
结果说明:最终执行了finally语句块。
示例三: 了解throws和throw的基本用法
throws是用于声明抛出的异常,而throw是用于抛出异常。
class MyException extends Exception {
public MyException() {}
public MyException(String msg) {
super(msg);
}
}
public class Demo3 {
public static void main(String[] args) {
try {
test();
} catch (MyException e) {
System.out.println("Catch My Exception");
e.printStackTrace();
}
}
public static void test() throws MyException{
try {
int i = 10/0;
System.out.println("i="+i);
} catch (ArithmeticException e) {
throw new MyException("This is MyException");
}
}
}
运行结果:
Catch My Exception
MyException: This is MyException
at Demo3.test(Demo3.java:24)
at Demo3.main(Demo3.java:13)
结果说明:
MyException是继承于Exception的子类。test()的try语句块中产生ArithmeticException异常(除数为0),并在catch中捕获该异常;接着抛出MyException异常。main()方法对test()中抛出的MyException进行捕获处理。
Java异常框架
Java异常架构图
1. Throwable
Throwable是 Java 语言中所有错误或异常的超类。
Throwable包含两个子类: Error 和 Exception。它们通常用于指示发生了异常情况。
Throwable包含了其线程创建时线程执行堆栈的快照,它提供了printStackTrace()等接口用于获取堆栈跟踪数据等信息。
2. Exception
Exception及其子类是 Throwable 的一种形式,它指出了合理的应用程序想要捕获的条件。
3. RuntimeException
RuntimeException是那些可能在 Java 虚拟机正常运行期间抛出的异常的超类。
编译器不会检查RuntimeException异常。例如,除数为零时,抛出ArithmeticException异常。RuntimeException是ArithmeticException的超类。当代码发生除数为零的情况时,倘若既"没有通过throws声明抛出ArithmeticException异常",也"没有通过try...catch...处理该异常",也能通过编译。这就是我们所说的"编译器不会检查RuntimeException异常"!
如果代码会产生RuntimeException异常,则需要通过修改代码进行避免。例如,若会发生除数为零的情况,则需要通过代码避免该情况的发生!
4. Error
和Exception一样,Error也是Throwable的子类。它用于指示合理的应用程序不应该试图捕获的严重问题,大多数这样的错误都是异常条件。
和RuntimeException一样,编译器也不会检查Error。
Java将可抛出(Throwable)的结构分为三种类型:被检查的异常(Checked Exception),运行时异常(RuntimeException)和错误(Error)。
(01) 运行时异常
定义: RuntimeException及其子类都被称为运行时异常。
特点: Java编译器不会检查它。也就是说,当程序中可能出现这类异常时,倘若既"没有通过throws声明抛出它",也"没有用try-catch语句捕获它",还是会编译通过。例如,除数为零时产生的ArithmeticException异常,数组越界时产生的IndexOutOfBoundsException异常,fail-fail机制产生的ConcurrentModificationException异常等,都属于运行时异常。
虽然Java编译器不会检查运行时异常,但是我们也可以通过throws进行声明抛出,也可以通过try-catch对它进行捕获处理。
如果产生运行时异常,则需要通过修改代码来进行避免。例如,若会发生除数为零的情况,则需要通过代码避免该情况的发生!
(02) 被检查的异常
定义: Exception类本身,以及Exception的子类中除了"运行时异常"之外的其它子类都属于被检查异常。
特点: Java编译器会检查它。此类异常,要么通过throws进行声明抛出,要么通过try-catch进行捕获处理,否则不能通过编译。例如,CloneNotSupportedException就属于被检查异常。当通过clone()接口去克隆一个对象,而该对象对应的类没有实现Cloneable接口,就会抛出CloneNotSupportedException异常。
被检查异常通常都是可以恢复的。
(03) 错误
定义: Error类及其子类。
特点: 和运行时异常一样,编译器也不会对错误进行检查。
当资源不足、约束失败、或是其它程序无法继续运行的条件发生时,就产生错误。程序本身无法修复这些错误的。例如,VirtualMachineError就属于错误。
按照Java惯例,我们是不应该是实现任何新的Error子类的!
对于上面的3种结构,我们在抛出异常或错误时,到底该哪一种?《Effective Java》中给出的建议是:对于可以恢复的条件使用被检查异常,对于程序错误使用运行时异常。
以上是关于[JAVA]异常的主要内容,如果未能解决你的问题,请参考以下文章