在程序设计和运行的过程中,出现问题是不可避免的,为此Java提供了异常处理机制来帮助程序员检查可能出现的异常情况,保证了程序的可读性和可维护性。
Error和Exception
Java程序的运行过程的异常情况中主要有两种:错误和异常,分别对应Java中的Error和Exception类,它们都继承了Throwable类。在Java中只有Throwable类型的实例才能被抛出(throw)或者捕获(catch),它是异常处理机制的基本组成类型。Java中对于错误和异常的处理是不同的,我们可以从异常中恢复程序但却不应该尝试从错误中恢复程序。
Error类描述了Java程序运行时系统的内部错误和资源耗尽错误,应用程序不应该抛出这种类型的对象。如果出现了这样的内部错误,除了通告给用户,并尽力使程序安全的终止外,再也没有其他的办法了,这种情况非常少见。
Exception又可以分为检查(checked)异常和不检查(unchecked)异常,检查异常必须在源代码里显示的进行捕获处理,这是编译期的一部分,例如IOException等。
不检查异常就是运行时异常,类似NullPointerException、ArrayIndexOutOfBoundsException,通常是编码过程中可以避免的逻辑错误,这种错误不应该出现,程序员应在编码的时候极力地避免这样的错误,不应该依赖于捕获处理。
捕获异常
Java中异常捕获结构由try、catch和finally 3部分组成。其中try语句块存放的是可能发生异常的Java语句;catch程序块在try语句块之后,用来处理被捕获的异常;finally语句块是异常处理结构的最后执行部分,无论try-catch语句块中的代码如何退出,都将执行finally语句块,通常用来关闭一些资源。
例如我们在使用JDBC操作数据库时,对异常的捕获处理:
// 声明jdbc对象
Connection conn = null;
PreparedStatement ps = null;
ResultSet rs = null;
// 声明变量
User u = null;
try {
// 加载驱动
Class.forName("com.mysql.cj.jdbc.Driver");
// 获取连接
conn = DriverManager.getConnection(
"jdbc:mysql://localhost:3306/db_database?useUnicode=true&characterEncoding=utf8&serverTimezone=GMT%2B8&useSSL=false",
"root", "123456");
// 创建sql命令
String sql = "Select * from tb_userlist where name = ? and pwd = ?";
// 创建sql命令对象
ps = conn.prepareStatement(sql);
// 给占位符赋值
ps.setString(1, name);
ps.setString(2, pwd);
// 执行sql
rs = ps.executeQuery();
// 遍历结果
while (rs.next()) {
u = new User();
u.setName(rs.getString("name"));
u.setPwd(rs.getString("pwd"));
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 关闭资源
try {
rs.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
try {
ps.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
try {
conn.close();
} catch (SQLException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
// 返回结果
return u;
注意:
1、try-catch代码段会产生额外的性能开销,它往往会影响JVM对代码进行优化,所以建议仅捕获有必要的代码段,尽量不要一个大的try包住整段代码;于此同时,也不应该利用异常控制代码流程。
2、尽量不要捕获类似Exception这样的通用异常,而是应该捕获特定异常,提高代码的可读性和可维护性。
3、不应该使用嵌套捕获,应该平行捕获。
4、try语句块中的局部变量的作用域仅局限于try语句块中,例如:
抛出异常
如果某个方法中可能会发生异常,但不想在当前方法处理或者当前方法无法处理时,应该使用throws、throw关键字在方法中抛出异常。
throws关键字通常被应用于声明方法时,用来指定方法可能抛出的异常。多个异常可使用逗号分隔。
throw关键字通常用于方法体中,用于抛出一个异常对象。程序在执行到throw语句时立即终止,它后面的语句都不执行。通过throw抛出异常后,如果想在上一级代码中来捕获并处理异常,则需要在抛出异常的方法使用throws关键字在方法的声明中指明要抛出的异常。