关于java中的try-catch-finally语句和return

本文详细探讨了函数返回语句与异常处理中的finally块之间的交互,提出了四种可行的返回语句放置策略,并通过测试代码展示了不同场景下的执行流程与结果。同时,强调了自定义异常类的使用原则与异常抛出的最佳实践。

原文链接:点击打开链接

本博客所有转载文章的所有权都归原作者所有,这里只是共享以及传播知识作用,每篇文章都会在开头标明出处,请尊重原作者版权。


第一:return语句并不是函数的最终出口,如果有finally语句,这在return之后还会执行finally(return的值会暂存在栈里面,等待finally执行后再返回)

第二:finally里面不建议放return语句,根据需要,return语句可以放在try和catch里面和函数的最后。可行的做法有四:

1.return语句只在函数最后出现一次。

2.return语句仅在try和catch里面出现。

3.return语句仅在try和函数的最后都出现。

4.return语句仅在catch和函数的最后都出现。

注意,除此之外的其他做法都是不可行的,编译器会报错。

 

(1)如果程序运行到try成功时可以返回结果,则采用方法2。(见下面的例子test0_1,在那个例子中,方法2和4都是可行的,但是推荐方法2?)

(2)如果程序运行到catch时(即中途出错时)无需再继续执行后面的代码了,则采取方法4;(见下面例子中的test0,在那个特殊的例子中,只能采取方法4)

(3)如果程序运行到try或catch时还需要继续执行后面的代码,则采取方法1(见下面的例子test0_2,该例子只能采用方法1)。

 

下面是测试代码:


public class Test {

    public static void main(String[] args) {
        System.out.println("=============test1==================");
        System.out.println(test1());
        System.out.println("===============================");

        System.out.println("=============test1_1==================");
        System.out.println(test1_1());
        System.out.println("===============================");

        System.out.println("\n============test2===================");
        System.out.println(test2());
        System.out.println("===============================");

        System.out.println("\n============test2_1===================");
        System.out.println(test2_1());
        System.out.println("===============================");

        System.out.println("\n============test3===================");
        System.out.println(test3());
        System.out.println("===============================");

        System.out.println("\n============test3_1===================");
        System.out.println(test3_1());
        System.out.println("===============================");
    }

    public static String test0() {
    	String a;
    	int b;
    	try{
    		b = 8/0;
    	}catch(Exception e){
    		return null;
    	}
    	a = String.valueOf(b);
    	return a+b;
    }
  
    public static String test0_1() {
    	String a;
    	int b;
    	try{
    		b = 8/0;
    		a = String.valueOf(b);
        	return a+b;
    	}catch(Exception e){
    		return null;
    	}
    	//return a+b;
    }
    
    public static String test0_2() {
    	String a;
    	int b=0;
    	try{
    		b = 8/0;
    	}catch(Exception e){
    	}
    	a = String.valueOf(b);
    	return a;
    }
    
    public static String test1() {
        String a = "in try";
        int n = -1;
        try{
            return a+n; //先执行这个,再执行finally
        } catch ( Exception e ) {

        } finally {
        	//对String和int的更改均无效
            a = "in finally";
            n = 123;
            System.out.println("do finally");
        }
        return a; //不会执行
    }  //总结出一点:return语句并不是函数的最终出口,如果有finally语句,这在return之后还会执行finally

    public static String test1_1() {
        String a = "in try";

        try{
            return a;
        } catch ( Exception e ) {

        } finally { //从eclpise报警告可看出,finally里面不建议有return语句
            a = "in finally";
            System.out.println("do finally");
            return a; //注释掉这句,eclipse将不再警告
        }
    }

    public static int test2() {
        int a = 1;

        try{
            return a;
        } catch ( Exception e ) {

        } finally {
            a = 2;
            System.out.println("do finally");
        }

        return a;
    } //很显然,finally里面更改无效,返回的是a=1

    public static int test2_1() {
        int a = 1;

        try{
            return a;
        } catch ( Exception e ) {

        } finally {
            a = 2;
            System.out.println("do finally");
            return a;
        }
    } //很显然,a取finally里面的值,a=2

    //Helper类,将整数转换成字符串
    static class Helper {
        int a;

        public String toString() {
            return String.valueOf(a);
        }
    }
    
    public static Helper test3() {
        Helper h = new Helper();
        h.a = 1;

        try{
            return h;
        } catch ( Exception e ) {

        } finally {
            h.a = 2; //对h.a的更改起作用!!
            		//因为在try里面返回的是一个句柄,它指向的对象的内容 是可以改变的
            System.out.println("do finally");
        }
        
        return h; //这个不会被执行
    }

    public static Helper test3_1() {
        Helper h = new Helper();
        h.a = 1;

        try{
            return h;
        } catch ( Exception e ) {

        } finally {
            h.a = 2; //返回a=2,这个不用说了
            System.out.println("do finally");
            return h;
        }
    }


    /**
     * 总结:
     * return语句,finally里面不建议放return语句,根据需要,可以放在try和catch里面
     * 
     */
    
}

异常处理之思考

    很少用自定义的异常类。

    因为对于一般的异常根本无需进行多么复杂的处理,所以用不着自定义的异常类)。如果有自定义的异常类,一般当场就catch掉而不抛出(抛出意味着什么?抛出一般会涉及到一个向上转型的过程,比如,为了能处理所有的异常,通常throws Exception,这样就会把所有的异常子类都向上转型为Exception父类,就失去了自定义异常类的意义)。

    但是实际上,对于异常,尽量做抛出处理,这样可以减少程序中到处都是try-catch。使用try-catch捕获异常时,建议一个try只放一行可能出现异常的代码,不要把很多无关的代码都写在try中(这是从编程的逻辑上而言,当多个异常集中在一起时,不妨只用一个try-catch。因为这样可以提高程序的效率,毕竟系统每处理一个try-catch语句都会有比较大的开销)


### Javatry-catch-finally语句的异常捕获与处理机制 在Java中,`try-catch-finally`结构是用于捕获处理程序运行时异常的重要工具。以下详细说明了该结构的工作原理及其关键特性。 #### 1. `try`块 `try`块用于包裹可能抛出异常的代码段。如果`try`块中的代码抛出了异常,则Java运行时系统会沿着方法调用栈逐层回溯,寻找能够处理该异常的代码[^1]。如果没有找到合适的异常处理代码,程序将终止并退出。 #### 2. `catch`块 `catch`块用于捕获由`try`块抛出的异常,并对其进行处理。每个`catch`块都必须指定它能够捕获的异常类型。当`try`块中的代码抛出一个异常时,Java运行时系统会检查`catch`块是否能够匹配该异常类型。如果匹配成功,则执行相应的`catch`块代码[^1]。 例如: ```java try { throw new Exception("我跑出的异常"); } catch (Exception e) { e.printStackTrace(); // 打印异常堆栈信息 } ``` 需要注意的是,如果发生的异常是`RuntimeException`类或其子类,则即使没有使用`try``catch`捕获,Java也能自动捕获这些异常,且编译通过。然而,这会导致程序在运行时终止[^3]。 #### 3. `finally`块 无论`try`块中是否发生异常,`finally`块中的代码都会被执行。这使得`finally`块成为释放资源或执行清理操作的理想位置。即使`try`或`catch`块中存在`return`语句,`finally`块仍然会被执行[^4]。 示例代码如下: ```java public static int test(String str) { try { Integer.parseInt(str); return 1; } catch (NumberFormatException e) { return -1; } finally { System.out.println("test结束"); // 无论是否有异常,此行代码都会执行 } } ``` #### 4. 异常传播 如果在`try`块中抛出的异常未被任何`catch`块捕获,则该异常会继续向上传播到调用栈中的上一层方法。这一过程称为异常传播。如果最终没有任何方法捕获该异常,则程序将终止[^1]。 #### 5. 多重`catch`块 可以为同一个`try`块定义多个`catch`块,以分别处理不同类型的异常。这些`catch`块按照定义顺序依次检查是否匹配抛出的异常类型。一旦某个`catch`块匹配成功并执行后,其余的`catch`块将被忽略[^1]。 示例代码如下: ```java try { // 可能抛出多种异常的代码 } catch (IOException e) { e.printStackTrace(); } catch (SQLException e) { e.printStackTrace(); } catch (Exception e) { e.printStackTrace(); } ``` #### 6. 异常分类处理 相比于使用`if-else`语句手动检测处理错误,`try-catch`结构具有显著优势。它避免了业务逻辑与错误处理代码的混合,从而提高了代码的可读性维护性。此外,`try-catch`结构能够确保所有异常都被适当捕获处理,而不会遗漏[^1]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值