在计算机程序运行的过程中,总是会出现各种各样的错误。
有一些错误是用户造成的,比如,希望用户输入一个int类型的年龄,但是用户的输入是abc

  1. //假设用户输入了abc
  2. String s = "abc";
  3. int n = Integer.parseInt(s);//numberFormatException

程序想要读写某个文件的内容,但是用户已经把它删除了

  1. String t = readFile("C"\\abc.txt");//FileNotFoundException

还有一些错误是随机出现,并且永远不可能避免的。

  • 网络突然断了,连接不到服务器了
  • 内存耗尽,程序崩溃了
  • 用户点“打印”,但是根本没有打印机

所以,一个健壮的程序必须处理各种各样的错误。
所谓错误,就是程序调用某个函数的时候,如果失败了,就表示出错。
调用方如何获调用失败的信息?有两种方法
方法一:约定返回错误码
例如,处理一个文件,如果返回0,表示成功,返回其他整数,表示约定的错误码

  1. int code = processFile("C:\\test.txt");
  2. if(code == 0) {
  3. //ok
  4. } else {
  5. //error
  6. switch (code){
  7. case 1:
  8. //file not found
  9. break;
  10. case 2:
  11. //no read permission
  12. break;
  13. default:
  14. //unknown error
  15. }
  16. }

因为使用int类型的错误码,想要处理就非常麻烦。这种方式常见于底层C函数。
方法二:在语言层面上提供一个异常处理机制。
Java内置了一套异常处理机制,总是使用异常来表示错误。
异常是一种class,因此它本身带有类型信息。异常可以在任何地方抛出,但只需要在上层捕获,这样就和方法调用分离。

  1. try {
  2. String s = processFile("C:\\test.txt");
  3. //ok
  4. } catch (FileNotFoundException e){
  5. //file not found
  6. } catch (SecurityException e){
  7. //no read permission
  8. } catch (IOException e){
  9. //io error
  10. } catch (Exception e){
  11. //other error
  12. }

因为Java的异常是class,它的继承关系如下
从继承关系可知:Throwable是异常体系的根,它继承自ObjectThrowable有两个体系:ErrorExceptionError表示严重的错误,程序对此一般无能为力,例如

  • OutOfMemoryError内存耗尽
  • NoClassDefFoundError无法加载某个Class
  • StackOverflowError 栈溢出

Exception则是运行时的错误,它可以被捕获并处理。

某些异常是应用程序逻辑处理的一部分,应该捕获并处理:

  • NumberFormatException:数值类型的格式错误
  • FileNotFoundException:未找到文件
  • SocketException:读取网络失败

还有一些异常是程序逻辑编写不对造成的,应该修复程序本身:

  • NullPointerException:对某个null的对象调用方法或字段
  • IndexOutOfBoundsException数组索引越界

Exception又分为两大类

  • RuntimeException以及它的子类
  • RuntimeException(包括IOExceptionReflectiveOperationException等等)

Java规定:

  • 必须捕获的异常,包括Exception及其子类,但不包括RuntimeException及其子类 ,这种类型的异常称为Checked Exception
  • 不需要捕获的异常,包括Error及其子类,RuntimeException及其子类。 :::danger 注意:编译器对RuntimeException及其子类不做强制捕获要求,不是指应用程序本身不应该捕获并处理RuntimeException。是否需要捕获,具体问题具体分析。 :::

    捕获异常

    捕获导演使用try...catch语句,把可能发生异常的代码放到try { ... }中,然后使用catch捕获对应的Exception及其子类:

    1. public class Main{
    2. public statc void main(String[] args){
    3. byte[] bs = toGBK("中文");
    4. System.out.println(Arrays.toString(bs));
    5. }
    6. static byte[] toGBK(String s ){
    7. try{
    8. //用指定的转换String 为byte[]
    9. return s.getBytes("GBK");
    10. } catch (UnsupportedEncodingException e) {
    11. //如果系统不支持GBK编码,会捕获到UnsupportedEncodingException
    12. System.out.println(e);//打印异常信息
    13. return s.getBytes();//尝试使用用默认编码
    14. }
    15. }
    16. }

    如果我们不捕获UnsupportedEncodingException,会出现编译失败的问题
    编译器会报错,错误信息类似
    java: 未报告的异常错误java.io.UnsupportedEncodingException; 必须对其进行捕获或声明以便抛出,并且准确地指出需要捕获的语句是return s.getBytes("GBK");。意思是这样的Checked Exception必须被捕获。
    这是因为String.getBytes(String)方法定义是

      public byte[] getBytes(String charsetName)
              throws UnsupportedEncodingException {
          if (charsetName == null) throw new NullPointerException();
          return StringCoding.encode(charsetName, value, 0, value.length);
      }
    

    在方法定义的时候,使用throws Xxx表示该方法可能抛出的异常类型。调用方在调用的时候,必须强制捕获这些异常,否则编译器会报错。
    toGBK()方法中,因为调用 了String.getBytes(String)方法, 就必须要捕获UnsupportedEncodingException。我们也可以不捕获它,而是在方法定义处用throw表示toGBK()方法可能会抛出异常,就可以让toGBK()方法通过编译器检查。

    public class Main{
      public static void main(String[] args){
          byte[] bs = toGBK("中文");
          System.out.println(Arrays.toString(bs));
      }
    
      static byte[] toGBK(String s ) throws UnsupportedEncodingException {
          return s.getBytes("GBK");
      }
    }
    

    上述代码仍然会得到编译错误,但这一次,编译器提示的还是调用reutrn s.getBytes("GBK");的问题了,而是byte[] bs = toGBK("中文");。因为在main()方法中,调用 toGBK(),没有捕获它声明的可能抛出的UnsupportedEncodingException.
    修复方法是在main()方法中捕获异常并处理
    可见,只要是方法声明的Checked Exception,不在调用层捕获,也必须在更高的调用层捕获。所有未捕获 的异常,最终也必须在main()方法中捕获,不会出现漏写try的情况。这是由编译器保证的。main()方法也是最后捕获Exception的机会。
    如果不想写任何try代码,可以直接把main()方法定义为throws Exception

    public class Main{
      public static void main(String[] args) throws Exception{
          byte[] bs = toGBK("中文");
          System.out.println(Arrays.toString(bs));
      }
      static byte[] toGBK(String s ) throws UnsupportedEncodingException{
          return s.getBytes("GBK");
      }
    }
    

    因为main()方法声明了可能抛出exception,也就声明了可能抛出所有的Exception,因此在内部就无需捕获了。代价就是一旦发生异常,程序会立刻退出。
    还有一些同学,喜欢在toGBK()内部“消化”异常

    static byte[] toGBK(String s ){
      try {
          return s.getBytes("GBK");
      } catch (UnsupportedException e) {
          //什么也不干
      }
      return null;
    }
    

    这种捕获后不处理的方式是非常不好的,即使真的什么也做不了,也要先把异常记录下来。

    static byte[] toGBK(String s ){
      try {
          return s.getBytes("GBK");
      } catch (UnsupportedException e) {
          e.printStackTrace();
      }
      return null;
    }
    

    所有异常都可以调用printStackTrace()方法打印异常栈,这是一个简单有用的快速打印异常的方法。
    小结:

  • Java使用异常来表示错误,并通过try ... catch捕获异常

  • Java的异常是class,并且从Throwable继承
  • Error是无需捕获的严重错误,Exception是应该捕获的可处理的错误
  • RuntimeException无需强制捕获,非RuntimeException(Checked Exception)需强制捕获,或者用throws声明
  • 不推荐捕获了异常但不进行任何处理。