最近遇到try-catch中return的问题,需要结合Java运行时内存空间才能够理解。下面结合几个例子,探讨其内存空间的变化。
例 1
public class java_try_catch {
public static void main(String[] args){
System.out.println(method());
}
public static int method(){
int a = 1;
try{
throw new Exception();
}catch(Exception e) {
System.out.println("catch : a = " + a);
return a;
}finally{
a = 2;
System.out.println("finally : a = " + a);
}
}
}
运行结果:
catch : a = 1
finally : a = 2
1
理解:
1、首先要理解 return 返回值 放在哪里。
方法 method() 中的a在 method 方法运行完之后就同该方法一起释放掉了,那return值怎么返回给主方法呢? 系统会分配一个临时空间(寄存器)用来存放method方法返回的值,然后再将这个临时值赋值给主方法的接收变量,再将临时空间(寄存器)释放 [参考http://bbs.csdn.net/topics/390891901 6楼的解释,再多的细节我也不清楚了= 。=]
2、理解上面代码的内存情况。
首先,变量 a 存放在栈中并赋值为1,try抛出异常,运行 catch 语句块。当运行到 return a 时,将 a 的值放入到寄存器,并执行 finally 语句块。而在 finally 中只是改变了栈中 a 的值,很显然寄存器中的值不会受到影响。因此,等 finally 语句块执行完后,主函数接收的是寄存器的值 1 。
例 2
import java.util.HashMap;
import java.util.Map;
public class java_try_catch {
public static void main(String[] args){
System.out.println(method1());
}
public static Map<String,Integer> method1() {
Map<String,Integer> map = new HashMap<String,Integer>();
map.put("value", 1);
try {
throw new Exception();
} catch(Exception e) {
System.out.println("catch : " + map);
return map;
} finally {
map.put("value", 2);
System.out.println("finally : " + map);
}
}
}
运行结果:
catch : {value=1}
finally : {value=2}
{value=2}
理解:
1、首先理解 对象是什么,引用是什么。
程序跟上面差不多,为什么结果就不一样了?这里我们要知道,对象是存放在堆中的,其引用存放在栈中,且该引用的值为对象在堆中的地址。即,对象是 new 关键字创建的,引用是指向对象的 “指针”。画图说明:
2、理解上面代码的内存情况。
首先,在堆空间创建 HashMap 对象并放入值 {value,1}。执行 try-catch 语句块,将 map 的值放入寄存器,注意到,map 是对象的引用,其存储的其实为对象的地址,因此,return 放到寄存器的值,也是这个地址值,即对象的地址。 当执行 finally 块时,map.put 修改的是对象的值。所以最终,主函数读取该地址得到的是 {value,2} 。
小结
- 网上讲了各种情况: return 在 try 中、catch 中、finally 中,return 是一个变量、引用等,会得到什么样的值,但其实理解了其内存情况,就不用在乎那么多东西,反而简单很多。
- 以上也只是我的理解,不足之处望多多指教。