程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> Bug分析之異常變量堆棧信息

Bug分析之異常變量堆棧信息

編輯:關於JAVA

異常是一種特殊的類,在創建異常時會保存創建時的方法調用堆棧鏡像。即,為了保留異常出現時的實時堆棧信息,不應復用異常,每個異常均需單獨new方式生成。

下面演示一段有問題的代碼並進行分析

1.問題代碼

a)自定義異常定義

  1. package demo.bce;
  2. public class MyException extends RuntimeException {
  3. private static final long serialVersionUID = -3802919537257556719L;
  4. private String id;
  5. public MyException(String id) {
  6. super();
  7. this.id = id;
  8. }
  9. public String getId() {
  10. return id;
  11. }
  12. public void setId(String id) {
  13. this.id = id;
  14. }
  15. @SuppressWarnings("unused")
  16. private MyException() {
  17. }
  18. }

b)自定義異常常量

  1. package demo.bce;
  2. public final class MyExceptionContext {
  3. // x1,x2,y1,y2的Throw相關堆棧信息在創建時一次性生成(不再變化)
  4. // 即使用此異常會得到錯誤的堆棧描述信息
  5. public static final MyException x1 = new MyException("X1");
  6. public static final MyException x2 = new MyException("X2");
  7. }

c)測試代碼

package demo.bce;

  1. public class MyMain {
  2. public static void main(String[] args) {
  3. testx();
  4. }
  5. // ///
  6. private static void testx() {
  7. try {
  8. x11();
  9. } catch (Exception e) {
  10. e.printStackTrace();
  11. }
  12. try {
  13. x12();
  14. } catch (Exception e) {
  15. e.printStackTrace();
  16. }
  17. try {
  18. x21();
  19. } catch (Exception e) {
  20. e.printStackTrace();
  21. }
  22. try {
  23. x22();
  24. } catch (Exception e) {
  25. e.printStackTrace();
  26. }
  27. }
  28. private static void x11() {
  29. throw MyExceptionContext.x1;
  30. }
  31. private static void x12() {
  32. throw MyExceptionContext.x2;
  33. }
  34. private static void x21() {
  35. throw MyExceptionContext.x1;
  36. }
  37. private static void x22() {
  38. throw MyExceptionContext.x2;
  39. }
  40. }

d)測試結果

  1. demo.bce.MyException
  2. at demo.bce.MyExceptionContext.<clinit>(MyExceptionContext.Java:7)
  3. at demo.bce.MyMain.x11(MyMain.Java:36)
  4. at demo.bce.MyMain.testx(MyMain.Java:14)
  5. at demo.bce.MyMain.main(MyMain.Java:7)
  6. demo.bce.MyException
  7. at demo.bce.MyExceptionContext.<clinit>(MyExceptionContext.Java:8)
  8. at demo.bce.MyMain.x11(MyMain.Java:36)
  9. at demo.bce.MyMain.testx(MyMain.Java:14)
  10. at demo.bce.MyMain.main(MyMain.Java:7)
  11. demo.bce.MyException
  12. at demo.bce.MyExceptionContext.<clinit>(MyExceptionContext.Java:7)
  13. at demo.bce.MyMain.x11(MyMain.Java:36)
  14. at demo.bce.MyMain.testx(MyMain.Java:14)
  15. at demo.bce.MyMain.main(MyMain.Java:7)
  16. demo.bce.MyException
  17. at demo.bce.MyExceptionContext.<clinit>(MyExceptionContext.Java:8)
  18. at demo.bce.MyMain.x11(MyMain.Java:36)
  19. at demo.bce.MyMain.testx(MyMain.Java:14)
  20. at demo.bce.MyMain.main(MyMain.Java:7)

代碼實際上在四個不同的方法中拋出了兩個不同的異常,但拋到四個異常的堆棧信息居然完全一致。

另外,x11和x21雖然拋同一個異常,但x11的異常無stackTrace,x21的異常有stackTrace信息。

2.代碼分析和猜想

在MyExceptionContext首次被調用時才生成常量異常x1和x2。注意x1和x2是同時生成的,且基本上處於相同的方法調用環境。故x1和x2的方法調用堆棧信息基本一致,進而在實際使用時嚴重誤導異常的拋出分析。

另外,通常情況下,異常是需要設置cause的。因此,也不應該嘗試常量異常(cause每次可能不一樣)。

3.簡單總結

使用異常時實時new一個出來返回以獲取正確方法調用堆棧信息。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved