异常及异常处理
一、常见异常
在 Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出);
Throwable: 有两个重要的子类:Exception(异常)和 Error(错误)
Error(错误):Java虚拟机无法解决的严重问题。如:JVM系统内部错误、资源耗尽等严重情况。比如:StackOverflowError和OOM。一般不编写针对性的代码进行处理。
**Exception(异常):**能被程序本身可以处理。
常见异常举例:
1.NullPointerException(空值针异常)
public void test1(){
// int[] arr = null;
// System.out.println(arr[3]);
String str = "abc";
str = null;
System.out.println(str.charAt(0));
}
2.NumberFormatException(数字格式化异常)
public void test4(){
String str = "123";
str = "abc";
int num = Integer.parseInt(str);
}
3.IndexOutOfBoundsException(角标越界异常)
public void test2(){
//ArrayIndexOutOfBoundsException 数组角标越界
// int[] arr = new int[10];
// System.out.println(arr[10]);
//StringIndexOutOfBoundsException 字符角标越界
String str = "abc";
System.out.println(str.charAt(3));
}
4.ArithmeticException (算数异常)
public void test6(){
int a = 10;
int b = 0;
System.out.println(a / b);
}
5.InputMismatchException (输入不匹配)
public void test5(){
Scanner scanner = new Scanner(System.in);
int score = scanner.nextInt();
System.out.println(score);
scanner.close();
}
6.ClassCastException (类型转化异常)
public void test3(){
Object obj = new Date();
String str = (String)obj;
}
二、异常处理
1.try-catch-finally
(1)格式:
try{
//可能出现异常的代码
}catch(异常类型1 变量名1){
//处理异常的方式1
}catch(异常类型2 变量名2){
//处理异常的方式2
}catch(异常类型3 变量名3){
//处理异常的方式3
}
finally{
//一定会执行的代码
}
(2)说明:
1. finally是可选的;
2.finally中声明的是一定会执行的代码,即使catch中出现异常,try中有return语句,catch中有return语句等情况;
3.像数据库连接、输入输出流、网络编程Socket等资源,JVM是不能自动的回收的,我们需要自己手动的进行资源的
释放。此时的资源释放,就需要声明在finally中;
3. 使用try将可能出现异常代码包装起来,在执行过程中,一旦出现异常,就会生成一个对应异常类的对象,根据此对象
的类型,去catch中进行匹配;
4. 一旦try中的异常对象匹配到某一个catch时,就进入catch中进行异常的处理。一旦处理完成,就跳出当前的 try-catch结构(在没有写finally的情况)。继续执行其后的代码;
5. catch中的异常类型如果没有子父类关系,则谁声明在上,谁声明在下无所谓,catch中的异常类型如果满足子父类关系,则要求子类一定声明在父类的上面。否则,报错;
6. 在try结构中声明的变量,再出了try结构以后,就不能再被调用;
7. try-catch-finally结构可以嵌套;
(3)举例说明:
@Test
public void testMethod(){
int num = method();
System.out.println(num);
}
public int method(){
try{
int[] arr = new int[10];
System.out.println(arr[10]);
return 1;
}catch(ArrayIndexOutOfBoundsException e){
e.printStackTrace();
return 2;
}finally{
System.out.println("我一定会被执行");
return 3;
}
}
运行结果:
2.throws+异常类型
注意:
-
"throws + 异常类型"写在方法的声明处。指明此方法执行时,可能会抛出的异常类型。一旦当方法体执行时,出现异常,仍会在异常代码处生成一个异常类的对象,此对象满足throws后异常
类型时,就会被抛出。异常代码后续的代码,就不再执行! -
try-catch-finally:真正的将异常给处理掉了。throws的方式只是将异常抛给了方法的调用者。 并没有真正将异常处理掉。 因此throws抛出异常之后还需要 try-catch-finally将异常进行处理。
举例说明:
public class ExceptionTest2 {
public static void main(String[] args){
try{
method2();
}catch(IOException e){
e.printStackTrace();
}
// method3();
}
public static void method3(){
try {
method2();
} catch (IOException e) {
e.printStackTrace();
}
}
public static void method2() throws IOException{
method1();
}
public static void method1() throws FileNotFoundException,IOException{
File file = new File("hello1.txt");
FileInputStream fis = new FileInputStream(file);
int data = fis.read();
while(data != -1){
System.out.print((char)data);
data = fis.read();//读操作
}
fis.close();//关闭相应流
System.out.println("hahaha!");
}
}
3.如何选择
开发中如何选择使用try-catch-finally 还是使用throws?
3.1 如果父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws,意味着如果子类重写的方法中有异常,必须使用try-catch-finally方式处理。
3.2 执行的方法a中,先后又调用了另外的几个方法,这几个方法是递进关系执行的。我们建议这几个方法使用throws的方式进行处理。而执行的方法a可以考虑使用try-catch-finally方式进行处理。
4.补充
(1)常用的异常对象处理的方式
① String getMessage() : 获取异常简单的描述信息
② printStackTrace():打印异常追踪的堆栈信息
(2)throw与throws区别
补充:throw格式举例说明
public class ReturnExceptionDemo {
static void methodA() {
try {
System.out.println("进入方法A");
throw new RuntimeException("制造异常");
} finally {
System.out.println("用A方法的finally");
}
}
static void methodB() {
try {
System.out.println("进入方法B");
return;
} finally {
System.out.println("调用B方法的finally");
}
}
public static void main(String[] args) {
try {
methodA();
} catch (Exception e) {
System.out.println(e.getMessage());
}
methodB();
}
}
运行结果:
进入方法A
用A方法的finally
制造异常
进入方法B
调用B方法的finally
throw表示抛出一个异常类的对象,生成异常类对象的过程。声明在方法体内
throws 属于异常处理的一种方式,声明在方法的声明处