throw和throws的区别以及try,catch,finally在有return的状况下执行的顺序

一,抛出异常有三种形式,一是throw,一个throws,还有一种系统自动抛异常。下面它们之间的异同。

(1)、系统自动抛异常

1.当程序语句出现一些逻辑错误、主义错误或类型转换错误时,系统会自动抛出异常:java

public static void main(String[] args) { 
    int a = 5, b =0; 
    System.out.println(5/b); 
    //function(); 
}

系统会自动抛出ArithmeticException异常。程序员

2.编程

public static void main(String[] args) { 
    String s = "abc"; 
    System.out.println(Double.parseDouble(s)); 
    //function(); 
}

系统会自动抛出NumberFormatException异常。函数

(2)、throw

throw是语句抛出一个异常,通常是在代码块的内部,当程序出现某种逻辑错误时由程序员主动抛出某种特定类型的异常测试

public static void main(String[] args) { 
    String s = "abc"; 
    if(s.equals("abc")) { 
      throw new NumberFormatException(); 
    } else { 
      System.out.println(s); 
    } 
    //function(); 
}

运行时,系统会抛出异常:spa

Exception in thread "main" java.lang.NumberFormatException .net

(3)、throws

throws是方法可能抛出异常的声明。(用在声明方法时,表示该方法可能要抛出异常)code

public void function() throws Exception{......}

当某个方法可能会抛出某种异常时用于throws 声明可能抛出的异常,而后交给上层调用它的方法程序处理orm

public class testThrows {
 
    public static void function() throws NumberFormatException {
        String s = "abc";
        System.out.println(Double.parseDouble(s));
    }
 
    public static void main(String[] args) {
        try {
            function();
        } catch (NumberFormatException e) {
            System.err.println("非数据类型不能强制类型转换。");
            //e.printStackTrace(); 
        }
    }
}

运行结果:对象

非数据类型不能强制类型转换。

(4)、throw与throws的比较
    throws出如今方法函数头;而throw出如今函数体。
    throws表示出现异常的一种可能性,并不必定会发生这些异常;throw则是抛出了异常,执行throw则必定抛出了某种异常对象。
    二者都是消极处理异常的方式(这里的消极并非说这种方式很差),只是抛出或者可能抛出异常,可是不会由函数去处理异常,真正的处理异常由函数的上层调用处理。

(5)、编程习惯 

    在写程序时,对可能会出现异常的部分一般要用try{...}catch{...}去捕捉它并对它进行处理;
    用try{...}catch{...}捕捉了异常以后必定要对在catch{...}中对其进行处理,那怕是最简单的一句输出语句,或栈输入e.printStackTrace();
    若是是捕捉IO输入输出流中的异常,必定要在try{...}catch{...}后加finally{...}把输入输出流关闭;
    若是在函数体内用throw抛出了某种异常,最好要在函数名中加throws抛异常声明,而后交给调用它的上层函数进行处理。

二,try,catch,finally在有return的状况下执行的顺序

结论:

一、无论有木有出现异常,finally块中代码都会执行;

二、当try和catch中有return时,finally仍然会执行;

三、finally是在return后面的表达式运算后执行的(此时并无返回运算后的值,而是先把要返回的值保存起来,管finally中的代码怎么样,返回的值都不会改变,任然是以前保存的值),因此函数返回值是在finally执行前肯定的;

四、finally中最好不要包含return,不然程序会提早退出,返回值不是try或catch中保存的返回值。

举例:

状况1:

try{} catch(){}finally{} return;

显然程序按顺序执行。

状况2:

try{ return; }catch(){} finally{} return;

程序执行try块中return以前(包括return语句中的表达式运算)代码;再执行finally块,最后执行try中return;finally块以后的语句return,由于程序在try中已经return因此再也不执行。
状况3:

try{ } catch(){return;} finally{} return;

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

try{ return; }catch(){} finally{return;}

 程序执行try块中return以前(包括return语句中的表达式运算)代码;再执行finally块,由于finally块中有return因此提早退出。
状况5:

try{} catch(){return;}finally{return;}

程序执行catch块中return以前(包括return语句中的表达式运算)代码;再执行finally块,由于finally块中有return因此提早退出
状况6:

try{ return;}catch(){return;} finally{return;}

程序执行try块中return以前(包括return语句中的表达式运算)代码;
  有异常:执行catch块中return以前(包括return语句中的表达式运算)代码;则再执行finally块,由于finally块中有return因此提早退出。
  无异常:则再执行finally块,由于finally块中有return因此提早退出。
最终结论

任何执行try 或者catch中的return语句以前,都会先执行finally语句,若是finally存在的话,若是finally中有return语句,那么程序就return了,因此finally中的return是必定会被return的,编译器把finally中的return实现为一个warning。

下面是个测试程序
public class FinallyTest  
{
    public static void main(String[] args) {
         
        System.out.println(new FinallyTest().test());;
    }

    static int test()
    {
        int x = 1;
        try
        {
            x++;
            return x;
        }
        finally
        {
            ++x;
        }
    }
}

结果是2。

分析:

在try语句中,在执行return语句时,要返回的结果已经准备好了,就在此时,程序转到finally执行了。 在转去以前,try中先把要返回的结果存放到不一样于x的局部变量中去,执行完finally以后,在从中取出返回结果, 所以,即便finally中对变量x进行了改变,可是不会影响返回结果。 它应该使用栈保存返回值。

参考博客:

https://blog.csdn.net/hjfcgt123/article/details/53349275

https://blog.csdn.net/kavensu/article/details/8067850

相关文章
相关标签/搜索