java异常处理(Exception handing)机制

一.异常:

1.1异常分类:

异常对象都是派生于throwable类的实例:

  • Error类说明java运行时存在内存错误或资源耗尽错误,出现这类错误,除了告诉用户,别无他法
  • Exception是需要关注的;它又分为RuntimeExceptionIOException

 

如果出现了RuntimeException,那么一定是你自己的问题

 

Error类或RuntimeException类的所有异常称为非受查(uncheck)异常,其他的异常成为受查(check)异常

 

1.2受查异常:

需要记住在以下情况中应该抛出异常:

  • 调用一个抛出受查异常的方法时,如:FileInputSteam构造器
  • 程序运行时发现错误,利用throw抛出一个受查异常
  • 程序出现错误时,如:数组越界(ArrayIndexOutOfBoundsException)
  • java虚拟机和运行时库内出现的内部错误

 

如果出现前两种异常之一,则必须告诉程序员调用这个方法可能会出现的异常,如果没有处理器捕获,当前执行的线程就会结束

对于可能被其他人调用的方法,应根据异常规范(exception specification),在方法首部声明者个可能的的异常:

class MyAnimation
{
    . . .
    public Image loadImage(String s) throws IOException
    {
        . . .
    }
}    

但是,无需声明java的内部错误,我们无法控制Error.

同样,不应该声明从RuntimeException继承的非受查异常:

class MyAnimation
{
    . . .
    void drawImage(int i) throws ArrayIndexOutOfBoundsException 
    {
        . . .
    }
}  

这些异常完全在我们的控制之下,与其去说明异常,我们更应该将精力花费在修改程序上

 

总之,一个方法必须声明可能抛出的受查异常,非受查异常要么是错误(Error),要么是可以避免发生的RuntimeException

 

注意:

  • 如果在子类中覆盖了一个父类的方法,子类声明的受查异常不可比父类的方法中声明的异常更为通用.
  • 如果父类方法没有抛出任何受查异常,那么子类也不能抛出任何受查异常

 

1.3抛出异常:

对于一个已知的异常类:

1).找到一个合适的异常类

2).创建这个类的对象

3).将对象抛出

 

String readData(Scanner in) throws EOFException
{
    . . .
    while (. . .)
    {
        if (!in.hasNext()) 
        {
            if (n < len)
            throw new EOFException();
        }
        . . .
    }
    return s;
}

 

一旦方法抛出了异常,该方法就不会返回给调用者,我们就不必再为返回的默认值或错误代码担忧  

 

 

1.4创建异常:

 

class FileFormatException extends IOException
{
    public FileFormatException() {}
    public FileFormatException(String gripe)
    {
        super(gripe);
    }
}

 

习惯上,定义的类包含两个构造器,一个是无参构造器,另一个是带有详细描述的构造器(父类Throwable的toString方法将会打印出详细信息)  

 

自定义构造器代码例:

String readData(BufferedReader in) throws FileFormatException
{
    . . .
    while (. . .)
    {
        if (ch == -1) 
        {
            if (n < len)
            throw new FileFormatException();
        }
        . . .
    }
    return s;
}

 

 

1.5.1捕获异常:

try
{
    code
    more code
    more code
}
catch (ExceptionType e)
{
    handler for this type
}

如果在try语句块中抛出了在catch语句块中说明的异常类,那么:

  • 程序将跳过try语句块其余的代码
  • 执行catch子句中的处理代码(如果在try中没有抛出任何异常,跳过catch子句)

如果方法中任何代码抛出了在catch中没有声明的异常,那么这个方法会立即退出

 

代码演示说明:

public void read(String filename)
{
    try
    {    
      InputStream in = new FileInputStream(filename);
      int b;
      while ((b = in.read()) != -1)
      {
          process input
      }
  }
  catch (IOException exception)
    {
        exception.printStackTrace();
    }
}

read方法可能抛出一个IOException,这将会跳出while循环,进入catch子句,并声称一个栈轨迹(stack trace)

通常,最好的办法是什么也不做,而是将异常传给调用者,让调用者去操心怎么做,如果采用这种方式,就必须声明这个方法可能会抛出的异常:public void read(String filename) throws IOException

 

 

1.5.2捕获多异常:

 

try
{
    code that might throw exceptions
}
catch (FileNotFoundException e)
{
    emergency action for missing files
}
catch (UnknownHostException e)
{
    emergency action for unknown hosts
}
catch (IOException e)
{
    emergency action for all other I/O problems
}

 

在JKD1.7之后,可以这样捕获异常:

catch (FileNotFoundException | UnknownHostException e)
{
    emergency action for missing files and unknown hosts
}

注意:捕获多个异常时,异常的变量为final

 

捕获多个异常不仅会使代码看起来更简洁,而且会使代码运行效率更快

 

1.6finally语句:

不管是否有异常被捕获,finally子句都必将被执行:

InputStream in = new FileInputStream(. . .);
try
{
    // 1
    code that might throw exceptions
    // 2
}
catch (IOException e)
{
    // 3
    show error message
    // 4
}
finally
{
    // 5
    in.close();
}
// 6

1).代码没有抛出异常.执行:1.2.5.6

2).抛出一个可以在catch中捕获的异常:

  • 如果catch没有抛出异常,执行:1.3.4.5.6
  • 如果catch子句抛出一个异常,执行:1.3.5(异常被抛会给方法调用者)

3).代码抛出一个异常,但没有catch可以捕获到,执行:1.5

 

因此,finally子句是无论如何都会执行的,下面的例子有一种令人意外的结果:

public static int f(int n)
{
    try
    {
        int r = n * n;
        return r;
    }
    finally
    {
        if (n == 2) return 0;
    }
}

调用f(2),try子句return结果r = 4,但是finally必然执行,return 0并覆盖 r = 4.

 

posted @ 2017-04-07 14:16  NOthingAJ  阅读(668)  评论(0编辑  收藏  举报