Java中try-catch-finally的一点理解

  在只有try-catch语句中,如果catch块中出现了return语句或者抛出了异常,那么catch之后的语句是执行不到的;但是如果将代码放入finally中,即使catch中出现了return语句或者抛出了异常,finally中的代码仍然是可以执行到的。看下面的程序:

public class Test {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.out.println(test());
	}

	public static int test() {
		int b = 23;
		try {
			System.out.println("yes");
			return b += 88;
		} catch (Exception e) {
			System.out.println("error:" + e);
		}
		System.out.println("next");
		return b;
	}
}

  因为try块中没有出现异常,且有return语句,所以后面的输出程序没有执行。但如果把输出程序放到finally中,则会执行到。

		try {
			System.out.println("yes");
			return b += 88;
		} catch (Exception e) {
			System.out.println("error:" + e);
		}finally{
			System.out.println("finally");
		}

  看第一段程序,最后有一个return b语句。如果缺少这条语句,程序会报错,提示test()方法必须返回一个int类型的值。可能有人会发现try语句中已经有了一个return语句了,为何还会报错呢。这是因为try块中放入的是可能出现异常的代码,如果发生了异常,系统就不会再执行try块中未执行的代码。所以存在一种情况就是程序还没执行到return语句时,就因为异常去执行catch中语句。这样的话,程序就跳过了return语句,所以必须在后面加上一个return。

  另外,如果我们把return语句放到finally块中,会有“finally does not complete normally”警告出现。原因可能如下:

1、不管try块、catch块中是否有return语句,finally块都会执行。
2、finally块中的return语句会覆盖前面的return语句(try块、catch块中的return语句),所以如果finally块中有return语句,Eclipse编译器会报警告“finally block does not complete normally”。

3、如果finally块中包含了return语句,即使前面的catch块重新抛出了异常,则调用该方法的语句也不会获得catch块重新抛出的异常,而是会得到finally块的返回值,并且不会捕获异常。

  下面来看finally和return的问题。如果我们在第一段程序中的finally中添加一些代码,如下所示:

package com.imooc1;

public class Test {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.out.println(test());
	}

	public static int test() {
		int b = 23;
		try {
			System.out.println("yes");
			return b += 88;
		} catch (Exception e) {
			System.out.println("error:" + e);
		} finally {
			if (b > 25) {
				System.out.println("b>25:" + b);
			}
			System.out.println("finally");
		}
		return b;
	}
}

  程序的运行结果会如何呢?这个主要是用来测试return和finally语句到底哪个先执行。在try语句中,返回了b=111; 在finally语句中判断是否大于25,如果大的话就输出大.最终结果如下:

yes
b>25:111
finally
111

  这说明finally语句是在try的return语句执行之后,return返回之前执行的

  如果把finally语句改成这样(会有上面所说的警告出现):

package com.imooc1;

public class Test {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.out.println(test());
	}

	public static int test() {
		int b = 23;
		try {
			System.out.println("yes");
			return b += 88;
		} catch (Exception e) {
			System.out.println("error:" + e);
		} finally {
			if (b > 25) {
				System.out.println("b>25:" + b);
			}
			System.out.println("finally");
			return 100;
		}
//		return b;
	}
}

  结果变为了:

yes
b>25:111
finally
100

  这说明finally块中的return语句会把try块中的return覆盖!

  如果把finally中语句变成这样:

package com.imooc1;

public class Test {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.out.println(test());
	}

	public static int test() {
		int b = 23;
		try {
			System.out.println("yes");
			return b += 88;
		} catch (Exception e) {
			System.out.println("error:" + e);
		} finally {
			if (b > 25) {
				System.out.println("b>25:" + b);
			}
			System.out.println("finally");
			b=100;
		}
		return b;
	}
}

   也就是在finally中改变b的值. 那么返回结果到底是原来的b值呢, 还是更改过的b值呢?结果如下:

yes
b>25:111
finally
111

  说明了一个问题,如果finally语句中没有返回语句覆盖的话,那么原来的返回值就不会变,不管你是不是改变了要返回的那个变量。但如果程序因为出现了异常而没有执行try中的return,则这样的改变是有效的。测试代码如下:

package com.imooc1;

public class Test {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		System.out.println(test());
	}

	public static int test() {
		int b = 23;
		int a = 0;
		try {
			System.out.println("b/a"+b/a);
			return b += 88;
		} catch (ArithmeticException e) {
			e.printStackTrace();
		} finally {
			if (b > 25) {
				System.out.println("b>25:" + b);
			}
			System.out.println("finally");
			b=100;
		}
		return b;
	}
}

  程序运行结果如下:

java.lang.ArithmeticException: / by zero
at com.imooc1.Test.test(Test.java:14)
at com.imooc1.Test.main(Test.java:7)
finally
100

时间: 2024-10-26 23:31:17

Java中try-catch-finally的一点理解的相关文章

【转】case: Java中try catch finally语句中含有return语句的执行情况(总结版)

Java中try catch finally语句中含有return语句的执行情况(总结版) 有一点可以肯定,finally块中的内容会先于try中的return语句执行,如果finall语句块中也有return语句的话,那么直接从finally中返回了,这也是不建议在finally中return的原因.下面来看这几种情况. 情况一(try中有return,finally中没有return): [java] view plain copy public class TryTest{ public 

Java中try,catch,finally的用法

Java中try,catch,finally的用法,以前感觉还算熟悉,但看到一篇博文才有更深点的理解,总结网友博客如下. Java异常处理的组合方式: 1.try+catch 运行流程:运行到try块中,如果有异常抛出,则转到catch块去处理.然后执行catch块后面的语句 2.try+catch+finally 运行流程:运行到try块中,如果有异常抛出,则转到catch块,catch块执行完毕后,执行finally块的代码,再执行finally块后面的代码. 如果没有异常抛出,执行完try

Java中关于WeakReference和WeakHashMap的理解

新美大的10月11日的笔试中有一道选择题,让选择函数返回结果,代码如下: 1 private static String test(){ 2 String a = new String("a"); 3 WeakReference<String> b = new WeakReference<String>(a); 4 WeakHashMap<String, Integer> weakMap = new WeakHashMap<String, In

java中try{}catch{}和finally{}的执行顺序问题

 今天我给大家讲解一下java的的错误和异常处理机制以及相关异常的执行顺序问题.如有不足的地方,欢迎批评指正~ 1.首相简单介绍一下java中的错误(Error)和异常(Exception) 错误和异常的介绍: 在java.lang软件包中有一个java.lang.Throwable类,这个类是java中所有错误和异常的超类. 在java中错误和异常的继承主要有两个: 分别为Error和Exception 这两个. Error:         是java中所有错误类的父类,就是jvm出现错误,

java中String、StringBuilder和StringBuffer理解

String.StringBuilder和StringBuffer理解 1>String java.lang.String 类 public final class String implements java.io.Serializable, Comparable<String>, CharSequence { /** The value is used for character storage. */ private final char value[]; /** Cache th

通过自己实现java中的ArrayList和LinkedList 深入理解数据结构---&quot;表&quot;

杂谈最基本数据结构--"表": 表结构是一种最基本的数据结构,最常见的实现是数组,几乎在每个程序每一种开发语言中都提供了数组这个顺序存储的线性表结构实现. 什么是线性表? 由0个或多个数据元素组成的有限序列.如果没有元素,称为空表,如果存在多个元素,则第一个元素无前驱,最后一个元素无后继,其他元素元素都有且只有一个前驱和后继. ArrayList和LinkedList ArrayList和LinkedList是顺序存储结构和链式存储结构的表在java语言中的实现. ArrayList提

java中try catch finally注意事项,finally在代码中什么时候运行

1.在java里函数抛出异常是需要在函数上定义的,除了runtimeException外 2.java中finally运行的位置在函数return前,其他的代码后.函数会运算完所有执行的代码,包括return里面的表达式,只是在return操作前去执行finally里面的代码.实例如下面的代码: public class Test{ public int add(int a,int b){ try { return a+b; } catch (Exception e) { System.out.

Java中==与equals的区别及理解

区别: “==” 比较的是两个引用在内存中指向的是不是同一对象(即同一内存空间),也就是说在内存空间中的存储位置是否一致.(引用类型) 如果两个对象的引用相同时(指向同一对象时),“==”操作符返回true,否则返回flase. 注:如果有对内存分配及变量存储位置(堆.栈.方法区常量池.方法区静态区)感兴趣的可以去看看这篇博客,里面写的很详细.对我还在学基础的人来说帮助很大,理解了很多内容,还有待消化. equals方法是由Object类提供的,可以由子类来进行重写 Object类默认的实现如下

java中的泛型的使用与理解

什么是泛型? 泛型是程序设计语言的一种特性.允许程序员在强类型程序设计语言中编写 体验泛型代码时定义一些可变部份,那些部份在使用前必须作出指明.各种程序设计语言和其编译器.运行环境对泛型的支持均不一样.将类型参数化以达到代码复用提高软件开发工作效率的一种数据类型.泛型类是引用类型,是堆对象,主要是引入了类型参数这个概念. 定义: 泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数.这种参数类型可以用在类.接口和方法的创建中,分别称为泛型类.泛型

java中get和set方法的理解与使用

之前看到java中经常出现一堆相对应的set和get,简单的知道set是传值get是取值. 例如: books.java 1 package test.testxml; 2 3 public class books { 4 private int id; 5 private String name; 6 private double price; 7 private String author; 8 9 10 public int getId() { 11 return id; 12 } 13