• Java异常之try,catch,finally,throw,throws


    Java异常之try,catch,finally,throw,throws

    你能区分异常和错误吗?

    我们每天上班,正常情况下可能30分钟就能到达。但是由于车多,人多,道路拥挤,致使我们要花费更多地时间,这就是生活中的异常!

    程序和生活一样都会出现异常,先来看个异常:

     

    上面出现的是算数错误的异常。

    在java中,除去一些编译上的错误(语法)之外,就有异常和错误!

    异常的定义是可自己进行处理后,程序依然可以正常运行下去!错误是Java虚拟机抛出的,终止程序的运行,这就是程序和异常的区别。

     

    一:什么是异常处理?

    异常处理机制就像我们对平时可能遇到的意外情况,预先想好了一些处理的办法。也就是说,在程序执行代码的时候,万一发生了异常,程序会按照预定的处理办法对异常进行处理,异常处理完毕后,程序继续运行。

    java的异常处理是通过5个关键字来实现的:try、catch、finally、throw、throws。

     

    二:java异常类的层次结构

    三.常见的异常类型

    Exception                                          异常层次结构的根类

    ArithmeticException                            算数错误情形

    ArrayIndexOutOfBoundsException       数组下标越界

    NullPointerException                           尝试访问null对象成员

    ClassNotFoundException                     不能加载所需的类

    InputMismatchException                     欲得到的数据类型与实际输入的类型不匹配

    IllegalArgumentException                    方法接受到非法参数

    ClassCastException                            对象强制类型转换出错

    NumberFormatException                     数字格式转换异常

     

    四.具体实例

    • try—catch
     1 package Test;
     2 
     3 import java.util.Scanner;
     4 
     5 public class Test_Test {
     6     public static void main(String[] args) {
     7         Scanner input =new Scanner(System.in);
     8         System.out.println("请输入被除数:");
     9         try {
    10             int num1=input.nextInt();
    11             System.out.println("请输入除数:");
    12             int num2=input.nextInt();
    13             System.out.println(String.format("%d / %d = %d",
    14                     num1, num2, num1 / num2));
    15         }catch (Exception e) {
    16             System.err.println("出现错误:被除数和除数必须是整数,"+
    17         "除数不能为零。");
    18             System.out.println(e.getMessage());
    19     }
    20 }

    运行结果如下:

    System.err.println();这种输出方式可以输出错误的消息,在控制台呈现红色。

    System.out用于正常的输出,也就是程序真正想输出的内容。而System.err用于出错信息的输出,也就是你本来不期待看到的东西。

    System.out.println(e.getMessage());

    这行的作用是——返回该错误的详细信息的字符串。

    • try-catch-finally
     1 package Test;
     2 
     3 import java.util.Scanner;
     4 
     5 public class Test_Test {
     6     public static void main(String[] args) {
     7         Scanner input =new Scanner(System.in);
     8         System.out.println("请输入被除数:");
     9         try {
    10             int num1=input.nextInt();
    11             System.out.println("请输入除数:");
    12             int num2=input.nextInt();
    13             System.out.println(String.format("%d / %d = %d",
    14                     num1, num2, num1 / num2));
    15         }catch (Exception e) {
    16             System.err.println("出现错误:被除数和除数必须是整数,"+
    17         "除数不能为零。");
    18             System.out.println(e.getMessage());
    19         }
    20             finally{
    21             System.out.println("Thanks");
    22         }
    23     }
    24 }

    运行结果如下:

    try-catch-finally 程序块的流程大致分为两种情况:

    1. 如果try块中所有语句正常执行完毕,那么finally块就会被执行。
    2. 如果try语句在执行过程中碰到异常,无论这种异常能否被catch块捕获到,都将执行finally块中的代码。

    try-catch-finally结构中try块是必须有的,catch和finally块为可选,但两者至少必须出现其中之一。

    • try—catch-catch-finally(多重catch块)
     1 package Test;
     2 
     3 import java.util.InputMismatchException;
     4 import java.util.Scanner;
     5 
     6 public class Test_Test {
     7     public static void main(String[] args) {
     8         Scanner input =new Scanner(System.in);
     9         System.out.println("请输入被除数:");
    10         try {
    11             int num1=input.nextInt();
    12             System.out.println("请输入除数:");
    13             int num2=input.nextInt();
    14             System.out.println(String.format("%d / %d = %d",
    15                     num1, num2, num1 / num2));
    16         }catch (InputMismatchException e) {
    17             System.err.println("被除数和除数必须是整数。");
    18         }
    19          catch (ArithmeticException e) {
    20                 System.err.println("除数不能为零。");
    21          }
    22         catch (Exception e) {
    23             System.err.println("其他未知异常。");
    24             System.out.println(e.getMessage());
    25         }
    26             finally{
    27             System.out.println("Thanks");
    28         }
    29     }
    30 }

    运行结果如下:

    所以,在写异常处理的时候,一定要把异常范围小的放在前面,范围大的放在后面,Exception这个异常的根类一定要刚在最后一个catch里面,如果放在前面或者中间,任何异常都会和Exception匹配的,就会报已捕获到...异常的错误。


    下面是try-catch-finally中包含return的情况:

    • 情况一:try{} catch(){}finally{} return;

                正常按程序顺序执行即可。

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 1;
    10         try
    11         {
    12             x++;
    13             System.out.println("我有用!");
    14         }
    15         catch (Exception e) {
    16             System.out.println("我没用!");
    17         }
    18         finally
    19         {
    20             ++x;
    21             System.out.println("我也有用!");
    22         }
    23         return 2;
    24     }
    25 }

    运行结果如下:

    • 情况2:try{ return; }catch(){} finally{} return;

              程序执行try块中return之前(包括return语句中的表达式运算)代码;
             再执行finally块,最后执行try中return;
             finally块之后的语句return,因为程序在try中已经return所以不再执行。

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 1;
    10         try
    11         {
    12             x++;
    13             System.out.println("我有用!");
    14             return 6;
    15         }
    16         catch (Exception e) {
    17             System.out.println("我没用!");
    18         }
    19         finally
    20         {
    21             ++x;
    22             System.out.println("我也有用!");
    23         }
    24         return 2;
    25     }
    26 }

    运行结果如下:

    • 情况3:try{} catch(){return;} finally{} return;

             程序先执行try,如果遇到异常执行catch块,
             有异常:则执行catch中return之前(包括return语句中的表达式运算)代码,再执行finally语句中全部代码,
                         最后执行catch块中return. finally之后也就是4处的代码不再执行。
             无异常:执行完try再finally再return.

    1.有异常的情况:

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 5;
    10         try
    11         {
    12             int num=x / 0;
    13             System.out.println(num);
    14         }
    15         catch (ArithmeticException e) {
    16             System.err.println("除数不能为0!");
    17             return 6;
    18         }
    19         finally
    20         {
    21             ++x;
    22             System.out.println("finally");
    23         }
    24         return 2;
    25     }
    26 }

    运行结果如下:

    2.无异常的情况:

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 5;
    10         try
    11         {
    12             System.out.println("try");
    13         }
    14         catch (ArithmeticException e) {
    15             System.err.println("除数不能为0!");
    16             return 6;
    17         }
    18         finally
    19         {
    20             ++x;
    21             System.out.println("finally");
    22         }
    23         return 2;
    24     }
    25 }

    运行结果如下:

    • 情况4:try{ return; }catch(){} finally{return;}

              程序执行try块中return之前(包括return语句中的表达式运算)代码;
              再执行finally块,因为finally块中有return所以提前退出。

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 5;
    10         try
    11         {
    12             int num = x / 0;
    13             System.out.println("try");
    14             return 3;
    15         }
    16         catch (ArithmeticException e) {
    17             System.err.println("除数不能为0!");
    18         }
    19         finally
    20         {
    21             ++x;
    22             System.out.println("finally");
    23             return 2;
    24         }
    25     }
    26 }

    运行结果如下:

    • 情况5:try{} catch(){return;}finally{return;}

              程序执行catch块中return之前(包括return语句中的表达式运算)代码;
              再执行finally块,因为finally块中有return所以提前退出。

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 5;
    10         try
    11         {
    12             int num = x / 0;
    13             System.out.println("try");
    14         }
    15         catch (ArithmeticException e) {
    16             System.err.println("除数不能为0!");
    17             return 4;
    18         }
    19         finally
    20         {
    21             ++x;
    22             System.out.println("finally");
    23             return 2;
    24         }
    25     }
    26 }

    运行结果如下:

    • 情况6:try{ return;}catch(){return;} finally{return;}

              程序执行try块中return之前(包括return语句中的表达式运算)代码;
              有异常:执行catch块中return之前(包括return语句中的表达式运算)代码;
                           则再执行finally块,因为finally块中有return所以提前退出。
              无异常:则再执行finally块,因为finally块中有return所以提前退出。

    1.有异常

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 5;
    10         try
    11         {
    12             int num = x / 0;
    13             System.out.println("try");
    14             return 4;
    15         }
    16         catch (ArithmeticException e) {
    17             System.err.println("除数不能为0!");
    18             return 4;
    19         }
    20         finally
    21         {
    22             ++x;
    23             System.out.println("finally");
    24             return 2;
    25         }
    26     }
    27 }

    运行结果如下:

    2.无异常

     1 package Test;
     2 
     3 public class Test_Test {
     4     public static void main(String[] args) {
     5         Test1();
     6     }
     7     
     8     public static int Test1(){
     9         int x = 5;
    10         try
    11         {
    12 //            int num = x / 0;
    13 //            System.out.println("try");
    14             return 4;
    15         }
    16         catch (ArithmeticException e) {
    17             System.err.println("除数不能为0!");
    18             return 4;
    19         }
    20         finally
    21         {
    22             ++x;
    23             System.out.println("finally");
    24             return 2;
    25         }
    26     }
    27 }

    运行结果如下:


    最终结论任何执行try 或者catch中的return语句之前,都会先执行finally语句,如果finally存在的话。
                      如果finally中有return语句,那么程序就return了,所以finally中的return是一定会被return的,
                      编译器把finally中的return实现为一个warning。


    • throw——抛出异常

    抛出异常有三种形式,一是throw,一个throws,还有一种系统自动抛异常。

    系统抛出异常:

    1 package Test;
    2 
    3 public class Test2 {
    4     public static void main(String[] args) {
    5         int a = 5, b =0;  
    6         System.out.println(5/b); 
    7     }
    8 
    9 }

    运行结果如下:

    throw抛出异常:

    throw是语句抛出一个异常。
    语法:throw (异常对象);

     1 package Test;
     2 
     3 public class Test2 {
     4     public static void main(String[] args) {
     5         String s = "abc";  
     6         if(s.equals("abc")) {  
     7             throw new NumberFormatException();
     8         } else {  
     9             System.out.println(s);  
    10         }  
    11     }
    12 
    13 }

    运行结果如下:

    • throws——声明异常

    throws是方法可能抛出异常的声明。(用在声明方法时,表示该方法可能要抛出异常)
    语法:[(修饰符)](返回值类型)(方法名)([参数列表])[throws(异常类)]{......}

     1 package Test;
     2 
     3 public class Test2 {
     4     public static void main(String[] args) {
     5         try {
     6             Test3();
     7         } catch (NumberFormatException e) {
     8             System.err.println("非数据类型不能转换。");
     9         }
    10     }
    11     
    12     public static void Test3() throws NumberFormatException{  
    13         String s = "abc";  
    14         System.out.println(Double.parseDouble(s));  
    15     }  
    16 }

    运行结果如下:

    如果在一个方法体中抛出了异常,那么我们就可以通过throws——声明异常来通知调用者,非常方便。

    throws表示出现异常的一种可能性,并不一定会发生这些异常;throw则是抛出了异常,执行throw则一定抛出了某种异常对象。

    后说一句,try-catch-finally虽好用,但是如果是滥用,这样只是会让程序的可读性变的很糟糕,当程序报错,就无法快速准确的定位了,物尽其用 人尽其才嘛!

  • 相关阅读:
    appcompat_v7 esvalues-v21 hemes_base.xml:158: error: Error: No resource
    Eclipse主题更换方法
    为什么一个RadioGroup增加子元素,循环不错误,不循环就错位
    关于RelativeLayout设置垂直居中对齐不起作用的问题
    Android 自定义 radiobutton
    thinkpad alert键一直处于按着的状态
    自定义RadioGrop,支持添加包裹着的RadioButton
    LyaoutParameters作用
    Java开发经验
    文章标题
  • 原文地址:https://www.cnblogs.com/wcf6676/p/4905909.html
Copyright © 2020-2023  润新知