在Java这种纯面向对象编程语言中,有着“一切皆对象”的说法。所以,java也是通过类来描述程序出现的异常现象。在编译的时候,一些严重的错误,比如说语法错误,会被虚拟机检测出来。而异常指的是程序运行过程中,受到了一些阻碍,无法继续进行下去。这个时候就会抛出异常,从当前位置向上级一层一层地抛出。
1、先看一个简单例子:
1 package myException;
2
3 public class Demo_1 {
4
5 public static void main(String[]args) {
6 Test test1=new Test();
7 int num = test1.div(2,0);
8 System.out.println(num);
9 System.out.println("over......");
10 }
11 }
12
13 class Test
14 {
15 public int div(int a, int b){16 return a / b;
17 }
18}
如果程序在进行整除运算时,不小心除了0,会发生什么现象?
抛出了算术异常,程序立刻终止,因为最后一句“over......”也没有打印出来。
2、try catch语句
既然有了异常,我们就要对其进行处理。
1 package myException;
2
3 public class Demo_1 {
4
5 public static void main(String[]args) {
6 Test test1=new Test();
7 try {
8 int num = test1.div(2,0);
9 System.out.println(num);
10 System.out.println("exception......");
11 }catch(ArithmeticException e) {
12 e.printStackTrace();
13 }
14 System.out.println("over......");
15 }
16 }
17
18 class Test
19 {
20 public int div(int a, int b){
21 return a / b;
22 }
23 }
发现:如果没有使用try catch语句,出现问题时,后面的代码都无法执行。如果加上try catch语句,那么语句块以外的还能够继续执行,但是语句块内部的代码,从出现问题的地方开始,都不能执行。
那么普通的代码块可以屏蔽这个错误,使得代码块外面的继续执行吗,见下面代码:
package myException;
public class Demo_1 {
public static void main(String[]args) throws ArithmeticException{
Test test1=new Test();
{
int num = test1.div(2,0);
System.out.println(num);
System.out.println("over......");
}
System.out.println("happy");
}
}
class Test
{
public int div(int a, int b) {
return a / b;
}
}
结果是不可以,看来只有 try catch语句块可以。
3、throw和throws
1 package myException;
2
3 public class Demo_1 {
4
5 public static void main(String[]args) throws ArithmeticException{
6 Test test1=new Test();
7 int num = test1.div(2,0);
8 System.out.println(num);
9 System.out.println("over......");
10 }
11 }
12
13 class Test
14 {
15 public int div(int a, int b) throws ArithmeticException{
16 if(b == 0)
17 throw new ArithmeticException();
18 return a / b;
19 }
20 }
基本的用法就是这样的,但是有几个注意点:
(1)throw方法体的内部,后面必须加切切实实new出来的对象,而throws用在方法名之后,不进入函数体,并且后面可以跟一个或多个类名(异常类),逗号隔开。
(2)如果一直向上抛出,而不catch,则会把异常抛给java处理机。
4、RuntimeException的特殊性
RuntimeException的子类有很多,比如上面的ArithmeticException,还有一些常见的比如IndexOutOfBoundsException(数组下标越界),NullPointerException(空指针异常)。为什么说它特殊呢?这种类型的异常通常都是由代码不当引起的,所以出现该异常时,希望程序停止,让维护人员进行代码的修改。所以在函数体内抛出的异常,函数名上是不需要申明的:
1 class Test
2 {
3 public int div(int a, int b) {
4 if(b == 0)
5 throw new ArithmeticException();
6 return a / b;
7 }
8 }
所以这样写也是合法的。当然,函数名后面加上“throws ArithmeticException”也是可以的。
如果是非RuntimeException或者其子类的,必须处理,或者逐层抛出再处理。比如进行文件操作,或者连接数据库的时候:
进行某些操作时,必须抛出异常,或者处理异常。比如上述代码如果这样写,编译都不会通过,eclipse报错“Unhandle Exception type ClassNotFoundException”。
5、Throwable类
在学习自定义异常类之前,先学习一下Throwable类。从字面上理解,“可以抛”的类,Throwable是所有异常类的基类,他的直接子类有Error类,Exception类。Throwable中的类主要是一些可以打印异常信息的方法,比如说 getMessage(), printStackTrace().他的子类可以根据具体情况覆盖重写这些方法。
6、自定义异常类
1 class DefineException extends Exception{
2 public DefineException() {
3
4 }
5 public DefineException(String str) {
6 super(str);
7 }
8 }
一般的格式就是这样,细节可以修改修改,可以根据实际情况覆盖重写Exception或者Throwable类的方法。当然,自定义的异常类必须要继承Throwable或者Exception。因为要想和throw、throws关键字搭配使用,必须是Throwable及其子类。
7、finally
finally一般和try catch相配合,在finally里处理一些必须要做、特别重要的信息,即使异常发生,finally里面的函数还是会被执行。特别是读写文件后的关闭、连接数据库的断开,比如下面的代码:
1 package com.ph;
2 import java.sql.Connection;
3 import java.sql.DriverManager;
4 import java.sql.PreparedStatement;
5 import java.sql.ResultSet;
6 public class Test1 {
7
8 public static void main(String[]args) {
9 PreparedStatement ps=null;
10 Connection ct=null;
11 ResultSet rs=null;
12 try {
13 //1.加载驱动
14 Class.forName("com.microsoft.sqlserver.jdbc.SQLServerDriver");
15 //2.得到链接 127.0.0.1:1433
16 ct=DriverManager.getConnection
17 ("jdbc:sqlserver://127.0.0.1:1433;databaseName=Mytest","sa","123456");
18
19 ps=ct.prepareStatement("select * from course");
20 rs=ps.executeQuery();
21 while(rs.next()) {
22 String cno=rs.getString(1);
23 String cname=rs.getString(2);
24 String tno=rs.getString(3);
25 System.out.println("cno "+cno+" cname "+cname+" tno "+tno);
26 }
27 }catch(Exception e) {
28 e.printStackTrace();
29 }finally {
30 try {
31 if(rs != null) rs.close();
32 if(ps != null) ps.close();
33 if(ct != null) ct.close();
34 }catch(Exception e) {
35 e.printStackTrace();
36 }
37 }
38 }
39 }
即使程序出错,也可以保证断开数据库连接,释放所占用的数据库资源,防止资源浪费。
仅作为学习笔记,如有错误,欢迎批评指正。