android常見異常總結

 

本文重在Java中異常機制的一些概念。寫本文的目的在於方便我很長時間後若是忘了這些東西可以通過這篇文章迅速回憶起來。
1. 異常機制
1.1 異常機制是指當程序出現錯誤後,程序如何處理。具體來說,異常機制提供了程序退出的安全通道。當出現錯誤後,程序執行的流程發生改變,程序的控制權轉移到異常處理器。
1.2 傳統的處理異常的辦法是,函數返回一個特殊的結果來表示出現異常(通常這個特殊結果是大家約定俗稱的),調用該函數的程序負責檢查並分析函數返回的結果。這樣做有如下的弊端:例如函數返回-1代表出現異常,但是如果函數確實要返回-1這個正確的值時就會出現混淆;可讀性降低,將程序代碼與處理異常的代碼混爹在一起;由調用函數的程序來分析錯誤,這就要求客戶程序員對庫函數有很深的瞭解。
1.3 異常處理的流程
1.3.1 遇到錯誤,方法立即結束,並不返回一個值;同時,拋出一個異常對象
1.3.2 調用該方法的程序也不會繼續執行下去,而是搜索一個可以處理該異常的異常處理器,並執行其中的代碼
異常的分類
2.1 異常的分類
2.1.1 異常的繼承結構:基類爲ThrowableErrorException繼承ThrowableRuntimeException和 IOException等繼承Exception,具體的RuntimeException繼承RuntimeException
2.1.2 ErrorRuntimeException及其子類成爲未檢查異常(unchecked),其它異常成爲已檢查異常(checked)。
2.2 每個類型的異常的特點
2.2.1 Error體系 Error類體系描述了Java運行系統中的內部錯誤以及資源耗盡的情形。應用程序不應該拋出這種類型的對象(一般是由虛擬機拋出)。如果出現這種錯誤,除了盡力使程序安全退出外,在其他方面是無能爲力的。所以,在進行程序設計時,應該更關注Exception體系。
2.2.2 Exception體系 Exception體系包括RuntimeException體系和其他非RuntimeException的體系
2.2.2.1 RuntimeException RuntimeException體系包括錯誤的類型轉換、數組越界訪問和試圖訪問空指針等等。處理RuntimeException的原則是:如果出現 RuntimeException,那麼一定是程序員的錯誤。例如,可以通過檢查數組下標和數組邊界來避免數組越界訪問異常。
2.2.2.2 其他(IOException等等)這類異常一般是外部錯誤,例如試圖從文件尾後讀取數據等,這並不是程序本身的錯誤,而是在應用環境中出現的外部錯誤。
2.3 C++異常分類的不同
2.3.1 其實,JavaRuntimeException這個類名起的並不恰當,因爲任何異常都是運行時出現的。(在編譯時出現的錯誤並不是異常,換句話說,異常就是爲了解決程序運行時出現的的錯誤)。
2.3.2 C++logic_errorJava中的RuntimeException是等價的,而runtime_errorJava中非RuntimeException類型的異常是等價的。
異常的使用方法
3.1 聲明方法拋出異常
3.1.1 語法:throws(略)
3.1.2 爲什麼要聲明方法拋出異常?方法是否拋出異常與方法返回值的類型一樣重要。假設方法拋出異常確沒有聲明該方法將拋出異常,那麼客戶程序員可以調用這個方法而且不用編寫處理異常的代碼。那麼,一旦出現異常,那麼這個異常就沒有合適的異常控制器來解決。
3.1.3 爲什麼拋出的異常一定是已檢查異常? RuntimeExceptionError可以在任何代碼中產生,它們不需要由程序員顯示的拋出,一旦出現錯誤,那麼相應的異常會被自動拋出。而已檢查異常是由程序員拋出的,這分爲兩種情況:客戶程序員調用會拋出異常的庫函數(庫函數的異常由庫程序員拋出);客戶程序員自己使用throw語句拋出異常。遇到Error,程序員一般是無能爲力的;遇到RuntimeException,那麼一定是程序存在邏輯錯誤,要對程序進行修改(相當於調試的一種方法);只有已檢查異常纔是程序員所關心的,程序應該且僅應該拋出或處理已檢查異常。
3.1.4 注意:覆蓋父類某方法的子類方法不能拋出比父類方法更多的異常,所以,有時設計父類的方法時會聲明拋出異常,但實際的實現方法的代碼卻並不拋出異常,這樣做的目的就是爲了方便子類方法覆蓋父類方法時可以拋出異常。
3.2 如何拋出異常
3.2.1 語法:throw(略)
3.2.2 拋出什麼異常?對於一個異常對象,真正有用的信息時異常的對象類型,而異常對象本身毫無意義。比如一個異常對象的類型是 ClassCastException,那麼這個類名就是唯一有用的信息。所以,在選擇拋出什麼異常時,最關鍵的就是選擇異常的類名能夠明確說明異常情況的類。
3.2.3 異常對象通常有兩種構造函數:一種是無參數的構造函數;另一種是帶一個字符串的構造函數,這個字符串將作爲這個異常對象除了類型名以外的額外說明。
3.2.4 創建自己的異常:當Java內置的異常都不能明確的說明異常情況的時候,需要創建自己的異常。需要注意的是,唯一有用的就是類型名這個信息,所以不要在異常類的設計上花費精力。
3.3 捕獲異常如果一個異常沒有被處理,那麼,對於一個非圖形界面的程序而言,該程序會被中止並輸出異常信息;對於一個圖形界面程序,也會輸出異常的信息,但是程序並不中止,而是返回用戶界面處理循環中。
3.3.1 語法:trycatchfinally(略)控制器模塊必須緊接在try塊後面。若擲出一個異常,異常控制機制會搜尋參數與異常類型相符的第一個控制器隨後它會進入那個catch 從句,並認爲異常已得到控制。一旦catch 從句結束對控制器的搜索也會停止。 3.3.1.1 捕獲多個異常(注意語法與捕獲的順序)(略)
3.3.1.2 finally的用法與異常處理流程(略)
3.3.2 異常處理做什麼?對於Java來說,由於有了垃圾收集,所以異常處理並不需要回收內存。但是依然有一些資源需要程序員來收集,比如文件、網絡連接和圖片等資源。
3.3.3 應該聲明方法拋出異常還是在方法中捕獲異常?原則:捕捉並處理哪些知道如何處理的異常,而傳遞哪些不知道如何處理的異常
3.3.4 再次拋出異常
3.3.4.1 爲什麼要再次拋出異常?在本級中,只能處理一部分內容,有些處理需要在更高一級的環境中完成,所以應該再次拋出異常。這樣可以使每級的異常處理器處理它能夠處理的異常。
3.3.4.2 異常處理流程對應與同一try塊的catch塊將被忽略,拋出的異常將進入更高的一級。
關於異常的其他問題
4.1 過度使用異常首先,使用異常很方便,所以程序員一般不再願意編寫處理錯誤的代碼,而僅僅是簡簡單單的拋出一個異常。這樣做是不對的,對於完全已知的錯誤,應該編寫處理這種錯誤的代碼,增加程序的魯棒性。另外,異常機制的效率很差。
4.2 將異常與普通錯誤區分開對於普通的完全一致的錯誤,應該編寫處理這種錯誤的代碼,增加程序的魯棒性。只有外部的不能確定和預知的運行時錯誤才需要使用異常。
4.3 異常對象中包含的信息一般情況下,異常對象唯一有用的信息就是類型信息。但使用異常帶字符串的構造函數時,這個字符串還可以作爲額外的信息。調用異常對象的getMessage()toString()或者printStackTrace()方法可以分別得到異常對象的額外信息、類名和調用堆棧的信息。並且後一種包含的信息是前一種的超集。

常用異常:

  UnsupportedOperationException不支持的操作

  IllegalArgumentException非法參數

  IndexOutOfBoundsException索引出界

  IllegalStateException非法狀態
異常跟普通的警告等有一定的區別。當應用程序發生異常時,會中斷正在執行的程序的正常指令流。也就是說,發生異常後面的代碼將得不到正確的執行。甚至還會觸發數據庫的回退操作。

Java開發平臺中,異常包括預定義異常與自定義異常。這兩種異常的類型互爲補充。作爲一個合格的程序開發人員,要善於在應用程序中使用異常。這可以提高應用程序的交互性。同時,也是保證應用程序正常運行的前提。故異常的處理對於開發一個優秀的應用程序來說非常的重要。爲此筆者認爲程序開發人員應該對 Java應用程序的常見異常有一個深入的瞭解。只有在瞭解這些常見異常的情況下,才能夠做好自定義異常的處理。

一、常見異常的類型與原因。

對於Java應用程序的常見異常,筆者認爲程序開發人員要從兩個方面去了解。一是要知道有哪些常見的Java應用程序異常,二是需要知道哪些原因可能會造成這個異常。這不僅需要程序管理人員在日常工作中要注意積累,在必要的情況下還需要去從其它渠道收集資料。筆者對此就進行一個分析,希望能夠對各位程序開發人員有一定的幫助。

1、 SQLException:操作數據庫異常類。

現在的Java應用程序大部分都是依賴於數據庫運行的。當Java應用程序與數據庫進行溝通時如果產生了錯誤,就會觸發這個類。同時會將數據庫的錯誤信息通過這個類顯示給用戶。也就是說,這個操作數據庫異常類是數據庫與用戶之間異常信息傳遞的橋樑。如現在用戶往系統中插入數據,而在數據庫中規定某個字段必須唯一。當用戶插入數據的時候,如果這個字段的值跟現有的紀錄重複了,違反了數據庫的唯一性約束,此時數據庫就會跑出一個異常信息。這個信息一般用戶可能看不到,因爲其發生在數據庫層面的。此時這個操作數據庫異常類就會捕捉到數據庫的這個異常信息,並將這個異常信息傳遞到前臺。如此的話,前臺用戶就可以根據這個異常信息來分析發生錯誤的原因。這就是這個操作數據庫異常類的主要用途。在Java應用程序中,所有數據庫操作發生異常時,都會觸發這一個類。所有此時Java應用程序本身的提示信息往往過於籠統,只是說與數據庫交互出現錯誤,沒有多大的參考價值。此時反而是數據庫的提示信息更加有使用價值。

2、 ClassCastException:數據類型轉換異常。

Java應用程序中,有時候需要對數據類型進行轉換。這個轉換包括顯示的轉換與隱式的轉換。不過無論怎麼轉換,都必須要符合一個前提的條件,即數據類型的兼容性。如果在數據轉換的過程中,違反了這個原則,那麼就會觸發數據類型轉換異常。如現在在應用程序中,開發人員需要將一個字符型的日期數據轉換爲數據庫所能夠接受的日期型數據,此時只需要在前臺應用程序中進行控制,一般不會有問題。但是,如果前臺應用程序缺乏相關的控制,如用戶在輸入日期的時候只輸入月、日信息,而沒有年份的信息。此時應用程序在進行數據類型轉換的時候,就會出現異常。根據筆者的經驗,數據類型轉換異常在應用程序開發中使一個出現的比較多的異常,也是一個比較低級的異常。因爲大部分情況下,都可以在應用程序窗口中對數據類型進行一些強制的控制。即在數據類型進行轉換之前,就保證數據類型的兼容性。如此的話,就不容易造成數據類型的轉換異常。如在只允許數值類型的字段中,可以設置不允許用戶輸入數值以外的字符。雖然說有了異常處理機制,可以保證應用程序不會被錯誤的運行。但是在實際開發中,還是要儘可能多的預見錯誤發生的原因,儘量避免異常的發生。

3、 NumberFormatException:字符串轉換爲數字類型時拋出的異常。

在數據類型轉換過程中,如果是字符型轉換爲數字型過程中出現的問題,對於這個異常在Java程序中採用了一個獨立的異常,即 NumberFormatException.如現在講字符型的數據“123456”轉換爲數值型數據時,是允許的。但是如果字符型數據中包含了非數字型的字符,如123#56,此時轉換爲數值型時就會出現異常。系統就會捕捉到這個異常,並進行處理。

Java應用程序中常見的異常類還有很多。如未找到相應類異常、不允許訪問某些類異常、文件已經結束異常、文件未找到異常、字段未找到異常等等。一般系統開發人員都可以根據這個異常名來判斷當前異常的類型。雖然不錯,但是好記性不如爛筆頭。程序開發人員在必要的時候(特別是存在自定義異常的時候),最後手頭有一份異常明細表。如此的話,無論是應用程序在調試過程中發現問題,還是運行過程中接到用戶的投訴,都可以及時的根據異常名字來找到異常發生的原因。從而可以在最短時間內解決異常,恢復應用程序的正常運行。這個措施筆者用了很多年,非常的有效。

二、異常管理的實用建議。

對於操作數據庫異常來說,Java應用程序只提供了一個異常類。故光憑Java應用程序的錯誤信息,往往不能夠幫助應用程序人員排除錯誤的原因。只能夠指名是應用程序錯誤還是數據庫錯誤導致的這個異常。爲了更進一步指明問題的原因,在數據庫層面定義異常的時候,最好能夠說明具體的原因。如前臺應用程序可能會調用數據庫的函數或者過程。此時在數據庫的函數或者過程中做好能夠說明某個異常的具體原因。如根據某個基礎表生成另一張表的時候,某個字段不能夠爲空等等。將這些異常信息說明清楚後,如果真的遇到類似的異常時,操作數據庫異常類就會將數據庫的異常信息反會給前臺用戶。從而有利於用戶尋找問題的原因,並在最短時間內改正。當然,這需要Java程序員與數據庫設計人員進行協調。

其次需要注意的是,異常並不是常態。也就是說,大部分異常可以通過前提的合理預見與預防,來消除。如設計到四則運算,可以在前臺應用程序窗口中限制在除數字段內輸入0值等手段來消除應用程序運行中可能產生的異常。不過這往往要求應用程序開發人員有比較豐富的工作經驗以及由比較嚴密的思維邏輯。雖然這有一定的難度,但是筆者認爲程序開發人員還是應該往這方面努力,而不要老是讓用戶作爲你的試驗品,讓用戶來發現應用程序中的設計Bug.筆者認爲,只有一些實在是程序人員無法控制的因素才允許拋出異常。如果應用程序開發人員能夠意識到這種錯誤、但是仍然沒有引起重視或者採取有效的措施防止出現這種異常,那麼筆者是不允許的。

 

ArithmeticException(除數爲0的異常), BufferOverflowException(緩衝區上溢異常), BufferUnderflowException(緩衝區下溢異常), IndexOutOfBoundsException(出界異常), NullPointerException(空指針異常), EmptyStackException(空棧異常), IllegalArgumentException(不合法的參數異常), NegativeArraySizeException, NoSuchElementException, SecurityException, SystemException, UndeclaredThrowableException

1. java.lang.NullPointerException
  異常的解釋是"程序遇上了空指針",簡單地說就是調用了未經初始化的對象或者是不存在的對象,即把數組的初始化和數組元素的初始化混淆起來了。數組的初始化是對數組分配需要的空間,而初始化後的數組,其中的元素並沒有實例化,依然是空的,所以還需要對每個元素都進行初始化(如果要調用的話)
  2. java.lang.ClassNotFoundException  異常的解釋是"指定的類不存在"
  3. java.lang.ArithmeticException  這個異常的解釋是"數學運算異常",比如程序中出現了除以零這樣的運算就會出這樣的異常。
  4. java.lang.ArrayIndexOutOfBoundsException
  異常的解釋是"數組下標越界",現在程序中大多都有對數組的操作,因此在調用數組的時候一定要認真檢查,看自己調用的下標是不是超出了數組的範圍,一般來說,顯示(即直接用常數當下標)調用不太容易出這樣的錯,但隱式(即用變量表示下標)調用就經常出錯了,還有一種情況,是程序中定義的數組的長度是通過某些特定方法決定的,不是事先聲明的,這個時候,最好先查看一下數組的length,以免出現這個異常。
  5. java.lang.IllegalArgumentException
  這個異常的解釋是"方法的參數錯誤",比如g.setColor(int red,int green,int blue)這個方法中的三個值,如果有超過255的也會出現這個異常,因此一旦發現這個異常,我們要做的,就是趕緊去檢查一下方法調用中的參數傳遞是不是出現了錯誤。
  6. java.lang.IllegalAccessException
  這個異常的解釋是"沒有訪問權限",當應用程序要調用一個類,但當前的方法即沒有對該類的訪問權限便會出現這個異常。對程序中用了Package的情況下要注意這個異常

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章