成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

java異常那些事

Kyxy / 2513人閱讀

摘要:一拋出異常發(fā)現(xiàn)錯(cuò)誤異常也是對(duì)象使用使用異常機(jī)制來報(bào)告錯(cuò)誤。異常也是普通的類類型。異常聲明中的語句執(zhí)行完成后會(huì)繼續(xù)執(zhí)行后的其他語句。非檢查異常拋出到上一級(jí)時(shí)可以不用進(jìn)行聲明合理的使用非檢查異??梢院喕a。

為什么需要異常機(jī)制:
Java的基本理念是“結(jié)構(gòu)不佳的代碼不能運(yùn)行” --- Java編程思想

最理想的是在編譯時(shí)期就發(fā)現(xiàn)錯(cuò)誤,但一些錯(cuò)誤要在運(yùn)行時(shí)才會(huì)暴露出來。對(duì)于這些錯(cuò)誤我們當(dāng)然不能置之不理。對(duì)于錯(cuò)誤而言的兩個(gè)關(guān)鍵是發(fā)現(xiàn)和處理錯(cuò)誤。Java提供了統(tǒng)一的異常機(jī)制來發(fā)現(xiàn)和處理錯(cuò)誤。

不考慮異常存在來看一下這個(gè)場景:

public void showObject(Object obj) {
    if (obj == null) {
        System.out.println("error obj is null");
    } else {
        System.out.println(obj.toString());
    }
}

對(duì)于showObject來說obj為null是一個(gè)錯(cuò)誤,要在輸出之前做錯(cuò)誤判斷,發(fā)生錯(cuò)誤的話把錯(cuò)誤打印出來作為錯(cuò)誤報(bào)告和處理。這里把錯(cuò)誤的發(fā)現(xiàn)、報(bào)告處理和正常業(yè)務(wù)邏輯放在了一起。但一些錯(cuò)誤往往比這復(fù)雜且不只一個(gè),如果我們?yōu)槊恳粋€(gè)錯(cuò)誤都去定義一個(gè)獨(dú)特錯(cuò)誤報(bào)告的形式且都將錯(cuò)誤處理代碼和正常業(yè)務(wù)代碼緊緊的耦合在一起那我們代碼會(huì)變得難以維護(hù)。

合理的使用異常不僅能使我們的代碼更加健壯,還能簡化開發(fā)提升開發(fā)效率。

一、拋出異常(發(fā)現(xiàn)錯(cuò)誤):
1、異常也是對(duì)象:

java使用使用異常機(jī)制來報(bào)告錯(cuò)誤。異常也是普通的類類型。Java自身已經(jīng)定義好了使用java時(shí)可能會(huì)產(chǎn)生的異常,使用java時(shí)java會(huì)自動(dòng)去檢查異常的發(fā)生,當(dāng)異常發(fā)生時(shí),java會(huì)自動(dòng)創(chuàng)建異常對(duì)象的實(shí)例并將其拋出。我們經(jīng)??吹降腘ullPointerException便是java已經(jīng)定義好的異常。

除了java自身定義的異常外我們可以自定義異常,但自定義的異常需要我們自己去檢查異常情形的發(fā)生,并自己創(chuàng)建異常對(duì)象和拋出。當(dāng)然也可以創(chuàng)建java自定義的異常并拋出,拋出異常使用throw關(guān)鍵字:

throw new Exception();

我們使用的第三方庫大多封裝了自己的異常,并在異常情形發(fā)生時(shí)將自定義異常通過throw拋出。所有的異常類型都繼承自Throwable類,所有Throwable類型的對(duì)象都可被拋出。

2、拋出異常:

異常發(fā)生,系統(tǒng)自動(dòng)創(chuàng)建異常實(shí)例并拋出,或我們自己創(chuàng)建異常實(shí)例拋出異常時(shí),代碼的正常執(zhí)行流程將會(huì)被終止,轉(zhuǎn)而去執(zhí)行異常處理代碼。

二、異常捕獲(處理錯(cuò)誤):
1、監(jiān)控區(qū)域:

當(dāng)異常拋出時(shí),自然拋出的異常應(yīng)該得到處理,這就需要將拋出的捕獲異常。但一個(gè)異常類型可能在很多地方被拋出,那么怎么去對(duì)特定的地方編寫特定的異常處理程序那?java采用一個(gè)最方便和合理的方式,即對(duì)可能產(chǎn)生異常的代碼區(qū)域進(jìn)行監(jiān)控,并在該區(qū)域后添加處理程序。

監(jiān)控的代碼區(qū)域放在try{}中,而異常處理代碼緊跟在try后的catch中:

try {
    /***/
} catch (ExceptionType e) {
    /*
    ***
    */
}

catch類似方法的申明括號(hào)中為異常的類型和該類類型的實(shí)例。表明當(dāng)前catch塊處理的是什么類型的異常,而e便是該異常拋出的實(shí)例對(duì)象。

當(dāng)try內(nèi)的代碼拋出異常時(shí),就會(huì)停止當(dāng)前流程,去匹配第一個(gè)catch中申明的異常類型與拋出類型相同的catch,如果匹配到則執(zhí)行其內(nèi)代碼。一個(gè)try中可能會(huì)拋出多種類型的異常,可以用多個(gè)catch去匹配。

2、捕獲所有異常:

注意catch中聲明的異常如果為當(dāng)前拋出異常的父類型也可以匹配。所以一般將基類的異常類型放在后面。

因?yàn)樗锌梢赃M(jìn)行捕獲的異常都繼承自Exception,所有可以catch中申明Exception類型的異常來捕獲所有異常,但最后將其放在最后防止將其他異常攔截了。

三、重新拋出異常
一、異常的兩種類型

先看一下異常的類層次結(jié)構(gòu)圖:

我們可以將異常分為檢查和非檢查兩種類型:

檢查異常:該類異常包含Exception及其子類異常,這些類型的異常的拋出必須有相應(yīng)的catch進(jìn)行捕獲,否則無法通過編譯。

非檢查異常:該類型的異常包含RuntimeException、Error和兩者的子類型,這類異??梢詻]有對(duì)應(yīng)的try-catch進(jìn)行捕獲也可通過編譯。當(dāng)異常發(fā)生時(shí)沒有相應(yīng)的捕獲則異常會(huì)自動(dòng)向上一級(jí)拋出,如此如果一直到main方法中還未被捕獲則會(huì)調(diào)用該異常的printStacjTrace方法輸出異常信息,并終止main的運(yùn)行。其中Error為Java自身錯(cuò)誤,這類錯(cuò)誤發(fā)生時(shí)我們并不能在業(yè)務(wù)代碼中去解決,如內(nèi)存不足,所以這類異常不需要去捕獲。

1、異常聲明:

catch中的語句執(zhí)行完成后會(huì)繼續(xù)執(zhí)行try-catch后的其他語句。所以當(dāng)try-catch后還有語句時(shí),一定要保證但異常發(fā)生時(shí)在catch中已經(jīng)對(duì)異常進(jìn)行了正確處理,后面的代碼可以得到正常的運(yùn)行,如果不能保證則應(yīng)該終止代碼向后的執(zhí)行或再次拋出異常。

一些異常在當(dāng)前的方法中不需要或無法進(jìn)行處理時(shí),可以將其拋出到上一層。要在方法中將異常拋出需要在方法中對(duì)要拋出的異常進(jìn)行聲明,這樣方法的調(diào)用者才能知道哪些異常可能會(huì)拋出,從而在調(diào)用方法時(shí)添加異常處理代碼。

非檢查異常拋出到上一級(jí)時(shí)可以不用進(jìn)行聲明,合理的使用非檢查異常可以簡化代碼。

(1) 異常聲明

在方法聲明的參數(shù)列表之后使用throws進(jìn)行異常聲明,多個(gè)異常類型使用逗號(hào)隔開:

void t () thrwos ExcptionTyep1, ExceptionType2 {
    
}

在方法中聲明了的異常在方法中可以不進(jìn)行捕獲,直接被拋出到上一級(jí)。異常聲明父類異常類型可以匹配子類異常類型,這樣當(dāng)有多個(gè)子類異常拋出時(shí),只用聲明一個(gè)父類異常即可,子類異常將被自動(dòng)轉(zhuǎn)換為父類型。

四、創(chuàng)建自定義異常:

要?jiǎng)?chuàng)建自己的異常必須得繼承自其它的異常,一般繼承Exception創(chuàng)建檢測異常,繼承RumtimeException創(chuàng)建非檢查異常。

一般情況下異常提供了默認(rèn)構(gòu)造器和一個(gè)接受String參數(shù)的構(gòu)造器。對(duì)于一般自定義的異常來說,只需要實(shí)現(xiàn)這兩個(gè)構(gòu)造方法就足夠了,因?yàn)槎x異常來說最有意義的是異常的類型,即異常類的名字,但當(dāng)異常發(fā)生時(shí)只需看到這個(gè)異常的類型就知道發(fā)生了什么,而其他一些操作在Throwable中已經(jīng)有定義。所以除非有一些特殊操作,不然在自定義異常時(shí)只需只需簡單的實(shí)現(xiàn)構(gòu)造方法即可。

五、異常信息:

所以異常的根類Throwable定義了我們需要的大多數(shù)方法:

// 獲取創(chuàng)建異常時(shí)傳入的字符串
String getMessage()
// 使用System.err輸出異常發(fā)生的調(diào)用棧軌跡
void printStackTrace()
// 使用傳入的PrintStream打印異常調(diào)用棧
void printStackTrace(PrintStream s)
// 使用PrintStreamOrWriter打印異常調(diào)用棧
void printStackTrace(PrintStreamOrWriter s)

獲取調(diào)用棧實(shí)例

StackTraceElement[] getStackTrace()

該方法放回StackTraceElement數(shù)組,StackTraceElement為調(diào)用方法棧的實(shí)例,改類型有以下常用方法:

// 返回棧代碼所在的文件名
String getFileName()
// 返回異常拋出地的行號(hào)
int getLineNumber()
// 返回棧的類名
String getClassName()
// 放回棧的方法名
String getMethodName()
六、異常鏈:
1、重新獲取異常

當(dāng)我們捕獲到一個(gè)異常時(shí)可能想將他在次拋出,但這樣直接拋出的話異常的棧信息是該異常原來的棧信息,不會(huì)是最新的再次拋出的異常的棧信息。如下:

class SimpleException extends Exception {
    public SimpleException() {
        
    }
    public SimpleException(String msg) {
        super(msg);
    }
}

public class Test {

    public void s() throws SimpleException {
        throw new SimpleException();
    }
    
    public void s2() throws SimpleException {
        try {
            s();
        } catch(SimpleException e) {
            throw e;
        }
    }
    
    public static void main(String[] args) {
        Test t = new Test();
        try {
            t.s2();
        } catch (SimpleException e) {
            e.printStackTrace();
        } 
    }
}

上面代碼輸出為:

com.ly.test.javatest.exceptiontest.SimpleException
    at com.ly.test.javatest.exceptiontest.Test.s(Test.java:19)
    at com.ly.test.javatest.exceptiontest.Test.s2(Test.java:24)
    at com.ly.test.javatest.exceptiontest.Test.main(Test.java:33)

可以看到異常拋出最終地為 com.ly.test.javatest.exceptiontest.Test.s(Test.java:19),但如果我們想讓異常拋出地變?yōu)閟2那?畢竟我們?cè)谶@里自己拋出了異常。
Thrwoable類的fillInStackTrac創(chuàng)建一個(gè)新的Throwable對(duì)象,并將當(dāng)前棧信息做新創(chuàng)建的Throwable異常的異常棧信息,然后返回。

2、異常鏈:

上面的做法又有另外一個(gè)問題,如果我們使用fillInStackTrace獲得新的異常,那原來的異常信息也就丟失了,如果我們想拋出新的異常當(dāng)又得包含原來的異常那?

Error、Exception和RuntimeException都含有一個(gè)接受Throwable對(duì)象的構(gòu)造方法,在創(chuàng)建新的異常時(shí)時(shí)傳入原來異常,即可保存原來異常。需要時(shí)使用getCause來獲取到。除了使用構(gòu)造方法傳入異常,還可使用initCase方法傳入異常。這其中的潛臺(tái)詞是“改異常是由什么異常造成的”。如下:

public class Test {

    public void s() throws Exception {
        throw new Exception();
    }
    
    public void s2() throws Exception {
        try {
            s();
        } catch(Exception e) {
            Exception ne = (Exception)e.fillInStackTrace();
            ne.initCause(e);
            throw ne;
        }
    }
    
    public static void main(String[] args) {
        Test t = new Test();
        try {
            t.s2();
        } catch (Exception e) {
            e.printStackTrace();
        } 
    }
}
六、總是執(zhí)行的finally

看一下面的代碼:

public class Test {
    public static void s() throws IOException {
        throw new IOException();
    }
    public static void main(String[] args) {
        String fileName = "C:	emp	est.txt";
        File file = new File(fileName);
        InputStream in = null;
        
        try {
            in = new FileInputStream(file);
            s();
            int tempbyte = in.read();
            in.close();
        } catch (IOException e) {
            if (in != null) {
                System.out.println("in");
            }
            e.printStackTrace();
        }
    }
}

可以看到要對(duì)in進(jìn)行close但正常的流程中發(fā)生了異常,導(dǎo)致正常流程中的in.close無法執(zhí)行,便跳到cattch中去執(zhí)行,上面于是又在catch中寫了一個(gè)關(guān)閉。著只是一個(gè)簡單清理操作,但如果需要執(zhí)行的清理操作不止一行而是非常多那?也是在正常流程和catch中寫兩遍嗎,這樣是非常不友好的,所以java提供了finally,如下

public class Test {
    public static void s() throws IOException {
        throw new IOException();
    }
    public static void main(String[] args) {
        String fileName = "C:	emp	est.txt";
        File file = new File(fileName);
        InputStream in = null;
        
        try {
            in = new FileInputStream(file);
            s();
            int tempbyte = in.read();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (in != null) {
                System.out.println("in");
            }
        }
    }
}

finally中的代碼無論異常是否發(fā)生都會(huì)被執(zhí)行,即使try中包含return語句,也會(huì)在放回之前執(zhí)行finally語句。

對(duì)于清理操作和一些異常發(fā)生也必得到執(zhí)行的代碼都應(yīng)該放到finally中。

清理未創(chuàng)建的資源:

上面介紹使用finally來釋放資源,但看下面這個(gè)情形:

public void test() {
    try {
        in = new BufferedReader(new FileReader());
        String s = in.readLine();
    } catch (FileNotFoundException e) {
        
    } catch (Exception e) {
        try {
            in.close();
        } catch (IOException e2) {
            System.out.println("in class false");
        }
    } finally {
        //in.close();
    }
}

這個(gè)例子可以看到如果new FileReader拋出了FileNotFoundException,那么in是不會(huì)被創(chuàng)建的,如果此時(shí)還在finally中執(zhí)行in.close()那么自然是行不同的。但如果拋出了IOExceptin異常,那么說明in成功創(chuàng)建但在readLine時(shí)發(fā)生錯(cuò),所以在catch中進(jìn)行close時(shí)in肯定已經(jīng)被創(chuàng)建。這種情形資源的釋放應(yīng)該放到catch中。

七、異常丟失:
1、在fianlly中return
public class Test {
    
    public static void main(String[] args) {
        try {
            int i = throwException();
            System.out.println(i);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static int throwException () throws Exception {
        try {
            throw new Exception();
        } catch (Exception e) {
            throw e;
        } finally {
            return 1;
        }
    }
}

上面代碼輸出:1

2、在finally中拋出異常
public class Test {
    
    public static void main(String[] args) {
        try {
            int i = throwException();
            System.out.println(i);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static int throwException () throws Exception {
        try {
            throw new Exception();
        } catch (Exception e) {
            throw e;
        } finally {
            throw new NullPointerException();
        }
    }
}

上面代碼輸出為:

java.lang.NullPointerException
    at com.ly.test.javatest.Test.throwException(Test.java:20)
    at com.ly.test.javatest.Test.main(Test.java:7)

可以看到main中捕獲到的是NullPointerException,首先拋出的Exception異常丟失了。

在開發(fā)中非特殊情形應(yīng)避免以上兩種情況的出現(xiàn)。

八、異常限制:

父類構(gòu)造器中聲明的異常在基類的構(gòu)造器中必須也聲明,因?yàn)楦割惖臉?gòu)造器總是會(huì)顯示會(huì)隱式(默認(rèn)構(gòu)造器)的被調(diào)用,而在子類構(gòu)造器中是無法捕獲父類異常的。但子類可以添加父類中沒有聲明的異常。

重載方法時(shí)子類只可拋出父類中聲明的異常,因?yàn)槲覀儠?huì)將子類對(duì)象去替換基類,這時(shí)如果重載的方法添加類新的異常聲明,那么原來的異常處理代碼將無法再正常工作。但子類方法可以減少或不拋出父類方法聲明的異常。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/68658.html

相關(guān)文章

  • Java 最困擾你的那些

    摘要:沒有操作符重載。最終類型在所有情況下應(yīng)該是默認(rèn)的,并用作為修飾符。這樣就會(huì)減少現(xiàn)在你會(huì)在和一些第三方的中見到的那些令人困惑的歷史遺留方法。在用過或是最新的之后你會(huì)覺得非常的繁瑣。這是最常見的關(guān)于的吐槽,但它這就是事實(shí)。 啊哈Reddit,沒了你我們還能在哪里從魚目混珠的網(wǎng)絡(luò)中提煉真正的精華?就在這雜亂無章的論壇中,的的確確存在著這樣一些精辟的討論。 比如有個(gè)叫Shambloroni的...

    Lorry_Lu 評(píng)論0 收藏0
  • 【面試】Java基礎(chǔ)中的那些-One

    摘要:前言面試中對(duì)于技術(shù)職位,一般分筆試與面談,如果面試官的一些小問題你可以立馬找到對(duì)應(yīng)的知識(shí)點(diǎn)擴(kuò)展開來,那么這就是你的優(yōu)勢,本系列將講述一些面試中的事,不會(huì)很詳細(xì),但是應(yīng)該比較全面吧。 前言 面試中對(duì)于技術(shù)職位,一般分筆試與面談,如果面試官的一些小問題你可以立馬找到對(duì)應(yīng)的知識(shí)點(diǎn)擴(kuò)展開來,那么這就是你的優(yōu)勢,本系列將講述一些java面試中的事,不會(huì)很詳細(xì),但是應(yīng)該比較全面吧。 主要內(nèi)容 說到...

    Stardustsky 評(píng)論0 收藏0
  • 談?wù)?em>Java引用和Threadlocal的那些

    摘要:容易導(dǎo)致內(nèi)存泄漏。如果我們的強(qiáng)引用不存在的話,那么就會(huì)被回收,也就是會(huì)出現(xiàn)我們沒被回收,被回收,導(dǎo)致永遠(yuǎn)存在,出現(xiàn)內(nèi)存泄漏。緩存行和一次定位,不會(huì)有沖突由于使用數(shù)組,不會(huì)出現(xiàn)回收,沒被回收的尷尬局面,所以避免了內(nèi)存泄漏。 1 背景 某一天在某一個(gè)群里面的某個(gè)群友突然提出了一個(gè)問題:threadlocal的key是虛引用,那么在threadlocal.get()的時(shí)候,發(fā)生GC之后,ke...

    justjavac 評(píng)論0 收藏0
  • JS異步那些 三 (Promise)

    摘要:異常處理異常處理一直是回調(diào)的難題,而提供了非常方便的方法在一次調(diào)用中,任何的環(huán)節(jié)發(fā)生,都可以在最終的中捕獲到錯(cuò)誤處理基本的小結(jié)具體的很多的用法可以參考阮一峰的入門教程,還有就是上面提到的電子書。 JS異步那些事 一 (基礎(chǔ)知識(shí))JS異步那些事 二 (分布式事件)JS異步那些事 三 (Promise)JS異步那些事 四(HTML 5 Web Workers)JS異步那些事 五 (異步腳本...

    wenzi 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<