解析Java中所有错误和异常的父类java.lang.Throwable

 更新时间:2016年03月13日 08:54:59   作者:Riddick  
这篇文章主要介绍了Java中所有错误和异常的父类java.lang.Throwable,文章中简单地分析了其源码,说明在代码注释中,需要的朋友可以参考下

在java语言中,错误类的基类是java.lang.Error,异常类的基类是java.lang.Exception。
1)相同点:java.lang.Error和java.lang.Exception都是java.lang.Throwable的子类,因此java.lang.Error和java.lang.Exception自身及其子类都可以作为throw的使用对象,如:throw new MyError();和throw new MyException();其中,MyError类是java.lang.Error的子类,MyException类是java.lang.Exception的子类。
2)不同点:java.lang.Error自身及其子类不需要try-catch语句的支持,可在任何时候将返回方法,如下面的方法定义:

public String myMethod() { 
throw new MyError(); 
} 

其中MyError类是java.lang.Error类的子类。
java.lang.Exception自身及其子类需要try-catch语句的支持,如下的方法定义是错误的:

public String myMethod() { 
throw new MyException(); 
} 

正确的方法定义如下:

public String myMethod() throws MyException { 
throw new MyException(); 
} 

其中MyException类是java.lang.Exception的子类。

JAVA异常是在java程序运行的时候遇到非正常的情况而创建的对象,它封装了异常信息,java异常的根类为java.lang.Throwable,整个类有两个直接子类java.lang.Error和java.lang.Exception.Error是程序本身无法恢复的严重错误.Exception则表示可以被程序捕获并处理的异常错误.JVM用方法调用栈来跟踪每个线程中一系列的方法调用过程,该栈保存了每个调用方法的本地信息.对于独立的JAVA程序,可以一直到该程序的main方法.当一个新方法被调用的时候,JVM把描述该方法的栈结构置入栈顶,位于栈顶的方法为正确执行的方法.当一个JAVA方法正常执行完毕,JVM回从调用栈中弹处该方法的栈结构,然后继续处理前一个方法.如果java方法在执行代码的过程中抛出异常,JVM必须找到能捕获异常的catch块代码.它首先查看当前方法是否存在这样的catch代码块,如果存在就执行该 catch代码块,否则JVM回调用栈中弹处该方法的栈结构,继续到前一个方法中查找合适的catch代码块.最后如果JVM向上追到了main()方法,也就是一直把异常抛给了main()方法,仍然没有找到该异常处理的代码块,该线程就会异常终止,如果该线程是主线程,应用程序也随之终止,此时 JVM将把异常直接抛给用户,在用户终端上会看到原始的异常信息.

Java.lang.throwable源代码解析

package java.lang; 
import java.io.*; 
/** 
* 
* Throwable是所有Error和Exceptiong的父类 
* 注意它有四个构造函数: 
* Throwable() 
* Throwable(String message) 
* Throwable(Throwable cause) 
* Throwable(String message, Throwable cause) 
* 
*/ 
public class Throwable implements Serializable { 
  private static final long serialVersionUID = -3042686055658047285L; 
 
  /** 
  * Native code saves some indication of the stack backtrace in this slot. 
  */ 
  private transient Object backtrace; 
 
  /** 
  * 描述此异常的信息 
  */ 
  private String detailMessage; 
 
  /** 
  * 表示当前异常由那个Throwable引起 
  * 如果为null表示此异常不是由其他Throwable引起的 
  * 如果此对象与自己相同,表明此异常的起因对象还没有被初始化 
  */ 
  private Throwable cause = this; 
 
  /** 
  * 描述异常轨迹的数组 
  */ 
  private StackTraceElement[] stackTrace; 
 
  /** 
  * 构造函数,起因对象没有被初始化可以在以后使用initCause进行初始化 
  * fillInStackTrace可以用来初始化它的异常轨迹的数组 
  */ 
  public Throwable() { 
   fillInStackTrace(); 
  } 
 
  /** 
  * 构造函数 
  */ 
  public Throwable(String message) { 
   //填充异常轨迹数组 
   fillInStackTrace(); 
   //初始化异常描述信息 
   detailMessage = message; 
  } 
 
  /** 
  * 构造函数,cause表示起因对象 
  */ 
  public Throwable(String message, Throwable cause) { 
   fillInStackTrace(); 
   detailMessage = message; 
   this.cause = cause; 
  } 
 
  /** 
  * 构造函数 
  */ 
  public Throwable(Throwable cause) { 
   fillInStackTrace(); 
   detailMessage = (cause==null ? null : cause.toString()); 
   this.cause = cause; 
  } 
 
  /** 
  * 获取详细信息 
  */ 
  public String getMessage() { 
   return detailMessage; 
  } 
 
  /** 
  * 获取详细信息 
  */ 
  public String getLocalizedMessage() { 
   return getMessage(); 
  } 
 
  /** 
  * 获取起因对象 
  */ 
  public Throwable getCause() { 
   return (cause==this ? null : cause); 
  } 
 
  /** 
  * 初始化起因对象,这个方法只能在未被初始化的情况下调用一次 
  */ 
  public synchronized Throwable initCause(Throwable cause) { 
   //如果不是未初始化状态则抛出异常 
   if (this.cause != this) 
    throw new IllegalStateException("Can't overwrite cause"); 
   
   //要设置的起因对象与自身相等则抛出异常 
   if (cause == this) 
    throw new IllegalArgumentException("Self-causation not permitted"); 
   
   //设置起因对象 
   this.cause = cause; 
   //返回设置的起因的对象 
   return this; 
  } 
 
  /** 
  * 字符串表示形式 
  */ 
  public String toString() {  
   String s = getClass().getName();   
   String message = getLocalizedMessage();  
   return (message != null) ? (s + ": " + message) : s; 
  } 
 
  /** 
  * 打印出错误轨迹 
  */ 
  public void printStackTrace() { 
   printStackTrace(System.err); 
  } 
 
  /** 
  * 打印出错误轨迹 
  */ 
  public void printStackTrace(PrintStream s) { 
   synchronized (s) { 
   //调用当前对象的toString方法 
    s.println(this); 
   //获取异常轨迹数组 
    StackTraceElement[] trace = getOurStackTrace(); 
    
   //打印出每个元素的字符串表示 
    for (int i=0; i < trace.length; i++) 
    s.println("\tat " + trace[i]); 
 
   //获取起因对象 
    Throwable ourCause = getCause(); 
    
   //递归的打印出起因对象的信息 
    if (ourCause != null) 
    ourCause.printStackTraceAsCause(s, trace); 
   } 
  } 
 
  /** 
  * 打印起因对象的信息 
  * @param s 打印的流 
  * @param causedTrace 有此对象引起的异常的异常轨迹 
  */ 
  private void printStackTraceAsCause(PrintStream s, 
           StackTraceElement[] causedTrace) 
  { 
   //获得当前的异常轨迹 
   StackTraceElement[] trace = getOurStackTrace(); 
   //m为当前异常轨迹数组的最后一个元素位置, 
   //n为当前对象引起的异常的异常轨迹数组的最后一个元素 
   int m = trace.length-1, n = causedTrace.length-1; 
   //分别从两个数组的后面做循环,如果相等则一直循环,直到不等或数组到头 
   while (m >= 0 && n >=0 && trace[m].equals(causedTrace[n])) { 
    m--; n--; 
   } 
   
   //相同的个数 
   int framesInCommon = trace.length - 1 - m; 
   
   //打印出不同的错误轨迹 
   s.println("Caused by: " + this); 
   for (int i=0; i <= m; i++) 
    s.println("\tat " + trace[i]); 
   //如果有相同的则打印出相同的个数 
   if (framesInCommon != 0) 
    s.println("\t... " + framesInCommon + " more"); 
 
   //获得此对象的起因对象,并递归打印出信息 
   Throwable ourCause = getCause(); 
   if (ourCause != null) 
    ourCause.printStackTraceAsCause(s, trace); 
  } 
 
  /** 
  * 打印出错误轨迹 
  */ 
  public void printStackTrace(PrintWriter s) { 
   synchronized (s) { 
    s.println(this); 
    StackTraceElement[] trace = getOurStackTrace(); 
    for (int i=0; i < trace.length; i++) 
     s.println("\tat " + trace[i]); 
 
    Throwable ourCause = getCause(); 
    if (ourCause != null) 
     ourCause.printStackTraceAsCause(s, trace); 
   } 
  } 
 
  /** 
  * 打印起因对象的信息 
  */ 
  private void printStackTraceAsCause(PrintWriter s, 
           StackTraceElement[] causedTrace) 
  { 
   // assert Thread.holdsLock(s); 
 
   // Compute number of frames in common between this and caused 
   StackTraceElement[] trace = getOurStackTrace(); 
   int m = trace.length-1, n = causedTrace.length-1; 
   while (m >= 0 && n >=0 && trace[m].equals(causedTrace[n])) { 
    m--; n--; 
   } 
   int framesInCommon = trace.length - 1 - m; 
 
   s.println("Caused by: " + this); 
   for (int i=0; i <= m; i++) 
    s.println("\tat " + trace[i]); 
   if (framesInCommon != 0) 
    s.println("\t... " + framesInCommon + " more"); 
 
   // Recurse if we have a cause 
   Throwable ourCause = getCause(); 
   if (ourCause != null) 
    ourCause.printStackTraceAsCause(s, trace); 
  } 
 
  /** 
  * 填充异常轨迹 
  */ 
  public synchronized native Throwable fillInStackTrace(); 
 
  /** 
  * 返回当前的异常轨迹的拷贝 
  */ 
  public StackTraceElement[] getStackTrace() { 
   return (StackTraceElement[]) getOurStackTrace().clone(); 
  } 
 
  
  /** 
  * 获取当前的异常轨迹 
  */ 
  private synchronized StackTraceElement[] getOurStackTrace() { 
   //如果第一次调用此方法则初始化异常轨迹数组 
   if (stackTrace == null) { 
   //获得异常轨迹深度 
    int depth = getStackTraceDepth(); 
   //创建新的异常轨迹数组,并填充它 
    stackTrace = new StackTraceElement[depth]; 
    
   for (int i=0; i < depth; i++) 
    stackTrace[i] = getStackTraceElement(i);//获取指定位标的异常轨迹 
   } 
   
   return stackTrace; 
  } 
 
  /** 
  * 设置异常轨迹 
  */ 
  public void setStackTrace(StackTraceElement[] stackTrace) { 
   //拷贝设置参数 
   StackTraceElement[] defensiveCopy = 
    (StackTraceElement[]) stackTrace.clone(); 
   
   //如果设置参数有空元素则抛出异常 
   for (int i = 0; i < defensiveCopy.length; i++) 
    if (defensiveCopy[i] == null) 
     throw new NullPointerException("stackTrace[" + i + "]"); 
 
   //设置当前对象的异常轨迹 
   this.stackTrace = defensiveCopy; 
  } 
 
  /** 
  * 异常轨迹的深度,0表示无法获得 
  */ 
  private native int getStackTraceDepth(); 
 
  /** 
  * 获取指定位标的异常轨迹 
  */ 
  private native StackTraceElement getStackTraceElement(int index); 
 
  
  private synchronized void writeObject(java.io.ObjectOutputStream s) 
   throws IOException 
  { 
   getOurStackTrace(); 
   s.defaultWriteObject(); 
  } 
} 

相关文章

  • iReport使用指南及常见功能示例详解

    iReport使用指南及常见功能示例详解

    这篇文章主要介绍了iReport使用指南及常见功能,本文以iReport 3.5.1为例,通过示例演示给大家介绍的非常详细,需要的朋友可以参考下
    2021-10-10
  • idea 与 maven 使用过程中遇到的问题及解决方案

    idea 与 maven 使用过程中遇到的问题及解决方案

    最近将IDEA 升级到 IntelliJ IDEA 2021.3.2,在将maven项目导入IDEA后,maven build时报异常,这个问题是IntelliJ IDEA 2021.3.2 不兼容导致的,下面小编给大家带来了idea 与 maven 使用过程中遇到的问题及解决方案,感兴趣的朋友一起看看吧
    2022-05-05
  • Java动态代理和反射机制详解

    Java动态代理和反射机制详解

    这篇文章主要介绍了Java动态代理和反射机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • java中对象和Map互相转换的几种常见方式举例

    java中对象和Map互相转换的几种常见方式举例

    Map在日常开发应用中的频率很高,最常用的实现类是HashMap和有序的TreeMap,下面这篇文章主要给大家介绍了关于java中对象和Map互相转换的几种常见方式举例,需要的朋友可以参考下
    2024-01-01
  • 快速入门介绍Java中强大的String.format()

    快速入门介绍Java中强大的String.format()

    这篇文章主要给大家介绍了如何快速入门介绍Java中强大的String.format()的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-03-03
  • Java垃圾回收机制的示例详解

    Java垃圾回收机制的示例详解

    本文主要围绕着Java垃圾回收当中的哪些内存需要回收?什么时候回收?如何回收?进行了详细讲解,感兴趣的小伙伴可以学习一下
    2022-04-04
  • 引入QQ邮箱发送验证码进行安全校验功能实现

    引入QQ邮箱发送验证码进行安全校验功能实现

    最近遇到这样的需求用户输入自己的邮箱,点击获取验证码,后台会发送一封邮件到对应邮箱中,怎么实现呢?下面小编给大家带来了引入QQ邮箱发送验证码进行安全校验功能,需要的朋友可以参考下
    2023-02-02
  • java中SynchronizedList和Vector的区别详解

    java中SynchronizedList和Vector的区别详解

    这篇文章主要介绍了java中SynchronizedList和Vector的区别详解,Vector是java.util包中的一个类。 SynchronizedList是java.util.Collections中的一个静态内部类。,需要的朋友可以参考下
    2019-06-06
  • 打包部署若依(RuoYi)SpringBoot后端和Vue前端图文教程

    打包部署若依(RuoYi)SpringBoot后端和Vue前端图文教程

    若依是一个使用Spring Boot作为后端和Vue.js作为前端的全栈应用开发平台,下面这篇文章主要给大家介绍了关于打包部署若依(RuoYi)SpringBoot后端和Vue前端的相关资料,需要的朋友可以参考下
    2024-05-05
  • java线程并发semaphore类示例

    java线程并发semaphore类示例

    Java 5.0里新加了4个协调线程间进程的同步装置,它们分别是Semaphore, CountDownLatch, CyclicBarrier和Exchanger,本例主要介绍Semaphore,Semaphore是用来管理一个资源池的工具,可以看成是个通行证
    2014-01-01

最新评论