定義
在Java語(yǔ)言中泞遗,將程序執(zhí)行中發(fā)生的不正常情況稱為“異常”(開(kāi)發(fā)過(guò)程中的語(yǔ)法錯(cuò)誤和邏輯錯(cuò)誤不是異常)
分類
- Error:Java虛擬機(jī)無(wú)法解決的嚴(yán)重問(wèn)題倔监,如JVM體統(tǒng)內(nèi)部錯(cuò)誤直砂,資源耗盡等嚴(yán)重情況。比如:StackOverflowError(棧溢出)和OOM(OutOfMemoryError堆溢出)浩习,一般不編寫針對(duì)性的代碼進(jìn)行處理
- Exception:其他因編程錯(cuò)誤或偶然的外在因素導(dǎo)致的一般性問(wèn)題静暂,它可以用針對(duì)性的代碼進(jìn)行處理,如:
1谱秽、 空指針訪問(wèn)
2洽蛀、試圖讀取不存在的文件
3、網(wǎng)絡(luò)連接中斷
4疟赊、數(shù)組角標(biāo)越界
java.lang.Throwable
- 編譯時(shí)異常(checked):IOException郊供、FileNotFountException
- 運(yùn)行時(shí)異常(unchecked):NullPointerException、ArrayIndexOutOfBoundsException近哟、ClassCastException驮审、NumberFormatException、InputMisMatchException吉执、ArithmaticException
package com.atguigu.java;
import java.util.Date;
import java.util.Scanner;
import org.junit.Test;
public class ExceptionTest {
//NullPointerException空指針異常
@Test
public void test1() {
String[] s1 = null;
System.out.println(s1[0]);
}
//ArrayIndexOutOfBoundsException數(shù)組角標(biāo)越界異常
@Test
public void test2() {
int[] array = new int[] {1,2,3};
System.out.println(array[3]);
}
//ClassCastException類型轉(zhuǎn)換異常
@Test
public void test3() {
Object obj = new Date();
String str = (String)obj;
}
//NumberFormatException
@Test
public void test4() {
String str = "abc";
int num = Integer.parseInt(str);
}
//InputMisMatchException輸入不匹配異常
@Test
public void test5() {
Scanner scanner = new Scanner(System.in);
int num = scanner.nextInt();//輸入"abc"
System.out.println(num);
}
//ArithmaticException算數(shù)異常
@Test
public void test6() {
int a = 10;
int b = 0;
System.out.println(a/b);
}
}
異常的處理:抓拋模型
- 過(guò)程一:拋疯淫,程序在正常運(yùn)行的過(guò)程中,一旦出現(xiàn)異常戳玫,就會(huì)在異常代碼處生成一個(gè)對(duì)應(yīng)異常類的對(duì)象熙掺,并將此對(duì)象拋出。一旦拋出對(duì)象后量九,其后代碼就不再執(zhí)行
- 過(guò)程二:抓适掰,可以理解為異常處理的方式颂碧。1、try-catch-finally(通常不針對(duì)運(yùn)行時(shí)異常編寫try-catch-finally类浪,針對(duì)編譯時(shí)異常载城,一定要考慮異常處理)。2费就、throws
try-catch-finally方式
說(shuō)明:
①:a)finally是可選的诉瓦。b)finally中聲明的是一定會(huì)被執(zhí)行的代碼,即使catch里面出現(xiàn)了異常力细,try中有return語(yǔ)句睬澡,catch中又return語(yǔ)句。c)像數(shù)據(jù)庫(kù)連接眠蚂,輸入輸出流煞聪,網(wǎng)絡(luò)編程Socket等資源,JVM是不能自動(dòng)的回收的逝慧,我們需要自己手動(dòng)進(jìn)行資源的釋放昔脯。此時(shí)的資源釋放,我們需要聲明在final里面
②:使用try將可能出現(xiàn)異常的代碼包裝起來(lái)笛臣,一旦出現(xiàn)異常云稚,就會(huì)生成一個(gè)對(duì)應(yīng)異常類的對(duì)象,根據(jù)此對(duì)象的類型沈堡,去catch中匹配
③:一旦try中的異常對(duì)象匹配到一個(gè)catch時(shí)静陈,就進(jìn)入catch進(jìn)行處理。一旦處理完成诞丽,就跳出當(dāng)前的try-catch結(jié)構(gòu)(在沒(méi)有寫finally的情況下)鲸拥,繼續(xù)執(zhí)行后面的代碼
④:catch中的異常類型如果滿足子父類關(guān)系,則要求子類一定聲明在父類上面率拒,否則報(bào)錯(cuò)
⑤:常用的兩種異常處理方式:一崩泡、String getMessage()
二、printStackTrace()
⑥:在try結(jié)構(gòu)中聲明的變量猬膨,出了try結(jié)構(gòu)角撞,就不能被調(diào)用
⑦:try-catch結(jié)構(gòu)可以嵌套
/*******************
try {
//可能出現(xiàn)異常的代碼
}catch(異常類型1 變量名1) {
//處理異常的方式1
}catch(異常類型2 變量名2) {
//處理異常的方式2
}catch(異常類型3 變量名3) {
//處理異常的方式3
}
...
finally {
//一定會(huì)執(zhí)行的代碼
}
********************************/
package com.atguigu.java;
import org.junit.Test;
public class ExceptionTest1 {
@Test
public void test() {
String str = "1234";
str = "abc";
try {
int num = Integer.parseInt(str);
}catch (NumberFormatException e) { //此時(shí)抓到NumberFormatException異常的對(duì)象
System.out.println("出現(xiàn)數(shù)值轉(zhuǎn)換異常");
}catch(NullPointerException e) {
System.out.println("出現(xiàn)空指針異常");
}
}
}
throws + 異常類型
throws +異常類型寫在方法的聲明處,指明此方法執(zhí)行時(shí)勃痴,可能會(huì)拋出的異常類型谒所,一旦當(dāng)方法體執(zhí)行時(shí),出現(xiàn)異常沛申,仍會(huì)在異常代碼處生成一個(gè)異常類的對(duì)象劣领,此對(duì)象滿足throws后面的異常類型,就會(huì)被拋出铁材,異常代碼后續(xù)的代碼尖淘,就不再執(zhí)行奕锌。
throws只是將異常拋給了方法的調(diào)用者,并沒(méi)有真正將異常處理掉
注意:子類重寫的方法拋出的異常不能比父類的更大村生。
手動(dòng)拋出異常對(duì)象
手動(dòng)生成一個(gè)異常惊暴,并拋出(throw)
package com.atguigu.java;
public class StudentTest {
public static void main(String[] args) {
try {
Student s1 = new Student();
s1.regist(-1);
} catch (Exception e) {
System.out.println(e.getMessage());
}
}
}
class Student{
private int id;
public void regist(int id) throws Exception {
if(id > 0) {
this.id = id;
}else {
//手動(dòng)拋出異常
// throw new RuntimeException("您輸入的數(shù)據(jù)非法");
throw new Exception("您輸入的數(shù)據(jù)非法");
}
}
}
用戶自定義異常類
- 如何自定義異常類
1、繼承現(xiàn)有的異常結(jié)構(gòu)趁桃,RuntimeException辽话、Exception
2、提供全局常量serialVersionUID
3卫病、提供重載的構(gòu)造器
package com.atguigu.java;
public class MyException extends RuntimeException {
static final long serialVersionUID = -7034897190745766939L;
public MyException() {
}
public MyException(String msg){
super(msg);
}
}