多线程之Throwable概述

Throw及其常见异常

Throwable是一个怎样的类?

 答:Throwable是java.lang包中一个专门用来处理异常的类。它有两个子类,即Error 和Exception,它们分别用来处理两组异常。 
Error用来处理程序运行环境方面的异常,比如,虚拟机错误、装载错误和连接错误,这类异常主要是和硬件有关的,而不是由程序本身抛出的。 
Exception是Throwable的一个主要子类。Exception下面还有子类,其中一部分子类分别对应于Java程序运行时常常遇到的各种异常的处理,其中包括隐式异常。比如,程序中除数为0引起的错误、数组下标越界错误等,这类异常也称为运行时异常,因为它们虽然是由程序本身引起的异常,但不是程序主动抛出的,而是在程序运行中产生的。Exception 子类下面的另一部分子类对应于Java程序中的非运行时异常的处理(在下图中将它们直接属于Exception了),这些异常也称为显式异常。它们都是在程序中用语句抛出、并且也是用语句进行捕获的,比如,文件没找到引起的异常、类没找到引起的异常等。 
一些主要子类对应的异常处理功能简要说明如下: 
ArithmeticException——由于除数为0引起的异常; 
ArrayStoreException——由于数组存储空间不够引起的异常; 
ClassCastException—一当把一个对象归为某个类,但实际上此对象并不是由这个类 创建的,也不是其子类创建的,则会引起异常; 
IllegalMonitorStateException——监控器状态出错引起的异常; 
NegativeArraySizeException—一数组长度是负数,则产生异常; 
NullPointerException—一程序试图访问一个空的数组中的元素或访问空的对象中的 方法或变量时产生异常; 
OutofMemoryException——用new语句创建对象时,如系统无法为其分配内存空 间则产生异常; 
SecurityException——由于访问了不应访问的指针,使安全性出问题而引起异常; 
IndexOutOfBoundsExcention——由于数组下标越界或字符串访问越界引起异常; 
IOException——由于文件未找到、未打开或者I/O操作不能进行而引起异常; 
ClassNotFoundException——未找到指定名字的类或接口引起异常; 
CloneNotSupportedException——一程序中的一个对象引用Object类的clone方法,但 此对象并没有连接Cloneable接口,从而引起异常; 
InterruptedException—一当一个线程处于等待状态时,另一个线程中断此线程,从 而引起异常,有关线程的内容,将在下一章讲述; 
NoSuchMethodException一所调用的方法未找到,引起异常; 
Illega1AccessExcePtion—一试图访问一个非public方法; 
StringIndexOutOfBoundsException——访问字符串序号越界,引起异常; 
ArrayIdexOutOfBoundsException—一访问数组元素下标越界,引起异常; 
NumberFormatException——字符的UTF代码数据格式有错引起异常; 
IllegalThreadException—一线程调用某个方法而所处状态不适当,引起异常; 
FileNotFoundException——未找到指定文件引起异常; 
EOFException——未完成输入操作即遇文件结束引起异常。

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();  
	      }  
	}  

课外小贴士~~~~~

多线程IO常见面试题
逻辑思考题: 1:你有一桶果冻,其中有黄色、绿色、红色三种,闭上眼睛抓取同种颜色的两个。抓取多少个就可以确定你肯定有两个同一颜色的果冻?

2:如果你有无穷多的水,一个3公升的提捅,一个5公升的提捅,两只提捅形状上下都不均匀,问你如何才能准确称出4公升的水?

3:有8个颜色,大小一样的小球,其中的一个质量和比其他的重,给你一个天平,你几次就可以称出那个重的球。(不要想着自己用手拿)

面试题:
1:请介绍一下自己
2:你有男/女朋友吗?你结婚了吗?
3:你有什么要问的吗?

面试题:

多线程:
1:多线程有几种实现方法,都是什么?同步有几种实现方法,都是什么?

2:启动一个线程是用run()还是start()?

3:sleep() 和 wait() 有什么区别?

4:线程的基本概念及线程的生命周期.

集合:
1: HashMap和Hashtable的区别。

2:Collection 和 Collections的区别。

3: List, Set, Map是否继承自Collection接口?

4:说出ArrayList,Vector, LinkedList的存储性能和特性?

5:你所知道的集合类都有哪些?主要方法?

IO:
1: java中有几种类型的流?JDK为每种类型的流提供了一些抽象类以供继承,请说出他们分别是哪些类?

2:什么是java序列化,如何实现java序列化?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吃掉你的脑子·

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值