zoukankan      html  css  js  c++  java
  • throw和throws的区别以及try,catch,finally在有return的情况下执行的顺序

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

    (1)、系统自动抛异常

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

    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(); 
    }

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

    Exception in thread "main" java.lang.NumberFormatException

    (3)、throws

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

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

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

    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的情况下执行的顺序

    结论:

    1、不管有木有出现异常,finally块中代码都会执行;

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

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

    4、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

  • 相关阅读:
    JSP源码、改写Servlet为JSP、查看转译成为Servlet的文件、JSP字符编码设置
    使用Tomcat部署应用
    缓冲与缓存
    过滤器及请求封装器实现字符编码
    过滤器与请求封装器实现字符替换
    过滤器
    与请求相关的监听器
    区块链简介
    https协议简介
    http协议简介
  • 原文地址:https://www.cnblogs.com/Tom-shushu/p/10244963.html
Copyright © 2011-2022 走看看