java 通过异常处理错误

java的基本理念是"结构不佳的代码不能够运行"

一.概念 

   发现错误的理想时机是编译阶段,然而,编译期间并不能找出所有的错误,余下的问题必须在运行时期解决。

二.基本异常

   异常情形(exceptional conditin)是指阻止当前方法或作用域继续执行的问题.

   普通问题是指在当前环境下能得到足够的信息,总能处理这个错误

   当抛出异常后,有几件事会随之发生。首先,同Java中其它对象的创建一样,将使用new在堆上创建异常对象。然后,当前的执行路径被终止,并且从当前环境中弹出对异常对象的引用。此时,异常处理机制接管程序,并开始寻找一个恰当的地方来继续执行程序。这个恰当的地方就是异常处理程序,它的任务是将程序从错误状态中恢复,以使程序要么换一种方式运行,要么继续运行下去。

if(t == null)
   throw new NullPointerException();//这样就把错误信息传播到更大的环境中,

三. 异常参数

    与使用java中的其它对象一样,我们总事用new再堆上创建异常对象,这也伴随着存储空间的分配和构造器的调用,所有标准异常都有两个构造器:一个是默认构造器,一个是接受字符串作为参数,以便能把有关信息放入异常对象的构造器;   

   关键字throw将产生许多有趣的结果,在使用new创建了异常对象之后,此对象的引用将传给throw.尽管返回的异常对象其类型通常与方法设计的返回类型不同,但从效果上看,它就像方法的返回,可以简单的把异常处理看成一种不同的返回机制,当然若过分的强调这种类比的化,就会有麻烦了.  另外还能用抛出异常的方式从当前的作用域退出,在这两种情况下,将会返回一个异常处理对象,然后退出方法或作用域

 抛出异常与方法正常返回值的相似到此为止,因为异常返回和普通方法返回的地点完全不同

  此外,能够抛出任意类型的Throwable对象,它是异常的根类,通常,对于不同的错误,要抛出相应的异常,错误处理信息可以保存在异常对象内或用异常类的名称来暗示.上异常环境通过这些信息来决定如何处理异常

 四. 捕获异常

    要明白异常时如何被捕获的,必须理解监控区域(guarded region)的概念,它时一段可能产生异常的代码,并且后面跟着处理这些异常的代码

Exception从基类Throwable继承的方法有

  • String getMessage(): 详细信息
  • String getLocalizedMessage(): 本地语言描述详细信息
  • void printStackTrace(): 调用栈显示了"把你带到异常抛出地点"的方法调用序列,输出到标准输出.
  • void printStackTrace(PrintStream): 调用栈显示了"把你带到异常抛出地点"的方法调用序列,输出到要输出的流.
  • void printStackTrace(java.io.PrintWriter): 调用栈显示了"把你带到异常抛出地点"的方法调用序列,输出到要输出的流.
  • Throwable fillInStackTrace(): 用于在Throwable对象的内部记录栈帧的当前状态.

栈轨迹:

printStackTrace()方法所提供的信息可以通过getStackTrace()方法直接访问.

getStackTrace()方法返回一个由根轨迹中的元素所构成的数组,每一个元素都表示栈中的一帧.

1.  try块

如果在方法内部抛出异常后,这个方法将在抛出异常的过程中结束,如果不想就此结束,可以设置一个try块来捕获异常

try{
   //Code that might generate exception
}

2.异常处理程序

    异常处理程序紧跟在try块之后,以关键字catch表示,异常处理程序必须紧跟在try块之后,当异常被抛出时,异常处理机制将负责搜寻参数与异常类型想匹配的第一个处理程序,然后进入catch子句执行,此时认为异常得到了处理,一旦catch子句结束才处理程序的查找过程结束,并不会在调用其它catch子句.

   异常处理模型

   终止模型:错误非常关键,一旦异常被抛出,即终止程序

   恢复模型:意思是异常处理程序的工作是修正错误,然后重新尝试调用出问题的方法,并认为第二次可以成功.如果要用java实现类似的恢复行为,那么在遇到错误时,就不能抛出异常,而时调用正确的方法来修正该错误,或者把try块放进一个while循环里,这样就不断进入while块,直到得到满意的结过

package object;

public class Cure {
    public static void main(String[] args) throws Exception
    {
        int i =5;
        while(i-->0)
        try {
                
        }catch(Exception e)
        {
            e.printStackTrace();
        }finally {
            System.out.println(i);
        }
    }
}

 

五. 创建自定义异常

 要自定义异常,必须从已有的异常类继承,最好是选择意思相近的异常类继承(不过这样的异常并不容易找),建立新的异常类型最简单的方法就是让编译器为你产生默认构造器,所以这几乎不用写多少代码:

package exceptions;
//: exceptions/InheritingExceptions.java
// Creating your own exceptions.

class SimpleException extends Exception {} //必须是throwable子类

public class InheritingExceptions {
  public void f() throws SimpleException {
    System.out.println("Throw SimpleException from f()");
    throw new SimpleException(); //这里抛出异常SimpleException
  }
  public static void main(String[] args) throws SimpleException{
    InheritingExceptions sed = new InheritingExceptions();
    try {
      sed.f();
    }catch(SimpleException e) //这里捕获异常SimpleException,并执行catch子句
    {
        System.out.println("caught it!");
    }
  }
} /* Output:
Throw SimpleException from f()
Caught it!
*///:~

编译器创建了默认构造器,它将自动调用基类的默认构造器,一般不会用Excetption(String)这样的构造器,这种构造器不实用,对异常来说最重要的是类名

  也可以通过写入System.err而将错误发送给标准错误流,通常这比System.out要好,因为System.out可以被重定向

package exceptions;
//: exceptions/FullConstructors.java

class MyException extends Exception {
  public MyException() {}
  public MyException(String msg) { super(msg); }//super调用类基类构造器,它结受一个字符串为参数
}

public class FullConstructors {
  public static void f() throws MyException {
    System.out.println("Throwing MyException from f()");
    throw new MyException();
  }
  public static void g() throws MyException {
    System.out.println("Throwing MyException from g()");
    throw new MyException("Originated in g()");
  }
  public static void main(String[] args) {
    try {
      f();
    } catch(MyException e) {
      e.printStackTrace(System.err);//将错误发送给标准错误流
//e.printStackTrace()//直接默认的话错误也是发送给标准错误
} try { g(); } catch(MyException e) { e.printStackTrace(System.out);//将错误发送到System.out ,System.out.可以重定向 } } } /* Output: Throwing MyException from f() exceptions.MyException Throwing MyException from g() exceptions.MyException: Originated in g() at exceptions.FullConstructors.f(FullConstructors.java:12) at exceptions.FullConstructors.main(FullConstructors.java:20) at exceptions.FullConstructors.g(FullConstructors.java:16) at exceptions.FullConstructors.main(FullConstructors.java:25) *///:~

 

posted @ 2019-01-19 18:44  江期玉  阅读(555)  评论(0编辑  收藏  举报