与大多数高级语言一样,Java通过return语句实现函数的带值返回功能。如:
public static String test(){
String str = "Hello world";
return str;
}
与C/C++的处理方式不同,Java在编写代码时,不允许return语句之后还是其它语句存在。如:
public static String test(){
String str = "Hello world";
return str;
str = "Hello world!"; // 编译错误。
}
但是这种语法规则又不是绝对的。如:
public static String test(){
String str = "try";
try {
return str;
}finally {
str = "finally";
}
}
此例编译可以通过,并且函数的返回值为字符串“try”。我们知道finally子句在try子句之后执行,上例从表面上看,“return str;”语句在“str = “finally”;”语句之前被执行。
如果Java语法不允许return语句之后存在其它语句,那么为什么上例在编译时能通过?
如果因为某种原因上例编译通过并能够运行,那么为什么这个函数会返回字符串“try”而不是“finally”呢?
-
finally子句总是被执行
有人会说,如果try子句中存在return语句的话,finally子句不会被调用。这种论断是错误的。如下例:
class Test {
public static void main(String[] args) throws Exception {
System.out.println(test());
}
public static String test() {
String str = "try";
try {
return str;
} finally {
System.out.println("finally called");
}
}
}
上例的调用结果如下:
try
其结果可以证明,无论try子句是否存在return语句,finally子句总是被调用。
-
return操作总是在最后执行
再看一下上文提到的函数:
public static String test(){
String str = "try";
try {
return str;
}finally {
str = "finally";
}
}
问题的实质在于
如果在try子句中存在return语句,JVM做了两件事:
1. 记住最后一个return语句所处代码位置上需要返回的变量值;
2. 将这个值放在函数调用过程的最后返回。
也就是说,上例中“return str;”一句,函数调用过程中会记录下此时str变量的值,这个值就是函数需要返回的值;当函数调用结束后,即finally子句结束后,将刚才记录下 来的值返回。
因此上述代码的返回值是“try”而不是“finally”。 其本质上可以用下面的代码表示:
public static String test(){
String tmp; // 声明一个“用于返回数据的变量”
String str = "try";
try {
tmp = str; // 将需要返回的值赋值给该变量
}finally {
str = "finally";
}
return tmp; // 将“用于返回数据的变量”的值返回
}
-
总是最后一个return语句在起作用
如果在try子句与finally子句中都有return语句,哪一个起作用呢?最后一个。如代码:
public static String test(){
String str = "try";
try {
return str;
}finally {
str = "finally";
return str;
}
}
上例函数的返回值为“finally”,因为是最后一个return语句在起作用。
其本质上可以用下面的代码表示:
public static String test(){
String tmp; // 声明一个“用于返回数据的变量”
String str = "try";
try {
tmp = str; // 将需要返回的值赋值给该变量
}finally {
str = "finally";
tmp = str; // 将需要返回的值赋值给该变量
}
return tmp; // 将“用于返回数据的变量”的值返回
}
-
return语句返回的是变量的值而不是对象的内容
关于变量的值与对象的内容之间的关系 ,这里不再多说。需要记住的是,return语句返回的是变量的值,与对象的内容无关。如:
public static StringBuilder test(){
StringBuilder build = new StringBuilder("try ");
try{
return build;
} finally {
build.append("finally");
build = new StringBuilder("new value");
}
}
上例中“return build;”一句记录下此时build变量的值(引用对象的地址值),之后第7句对build变量的重新赋值已经无法影响函数的返回值,但是第6句修改 了build变量所指对象的内容。因此上例的返回值,其所指对象的内容为“try finally”而不是另一个内容为“new value”的新对象。
上述代码本质上可以用下面的代码表示:
1. public static StringBuilder test(){
2. StringBuilder tmp; // 声明一个“用于返回数据的变量”
3. StringBuilder build = new StringBuilder("try ");
4. try{
5. tmp = build; // 将需要返回的值赋值给该变量
6. } finally {
7. build.append("finally");
8. build = new StringBuilder("new value");
9. }
10. return tmp; // 将“用于返回数据的变量”的值返回
11. }
package com.wangdi.tryfinally;
/**
* @author gstarwd
*/
public class TryFinnally {
public static void main(String[] args) {
System.out.println(new TryFinnally().returnStringMethod());
System.out.println(new TryFinnally().returnIntMethod());
System.out.println(new TryFinnally().returnPersonMethod().getName());
}
private String returnStringMethod() {
String returnStr = "initail";
try {
returnStr = "inside try";
return returnStr;
} finally {
returnStr = "inside finally";
}
}
private int returnIntMethod() {
int returnInt = 1;
try {
returnInt = 2;
return returnInt;
} finally {
returnInt = 3;
}
}
private Person returnPersonMethod() {
Person returnPerson = new Person();
returnPerson.setName("initail");
try {
returnPerson.setName("inside try");
return returnPerson;
} finally {
returnPerson.setName("inside finally");
}
}
}
class Person {
Person() {
}
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}