最近一道面试题引起了笔者的兴趣,在try-catch-finnally里面分别添加了一个return,那么返回的值是哪个里面的。这种问题很纠结啊,单纯按照书本知识肯定会回答,finnally里面的哈~可是实际情况又是复杂多变的。
第一种情况(try里有return)
import java.util.concurrent.locks.ReentrantLock;
/**
* @author hassop
* @Description
* @date 2016/8/20 0019
* To change this template use File | Settings | File Templates.
*/
public class TryTest {
private final static ReentrantLock clientLock = new ReentrantLock();
public static final String test() {
String t = "";
try {
clientLock.lock();
t = "try";
System.out.println("加锁成功");
return t;
} catch (Exception e) {
// result = "catch";
t = "catch";
return t;
} finally {
t = "finally";
clientLock.unlock();
System.out.println("解锁成功");
/*return t;*/ //加上返回finnally
}
}
public static void main(String[] args) {
System.out.println(TryTest.test());
}
}
执行结果:
加锁成功
解锁成功
try
第二种情况(finally里有return)
import java.util.concurrent.locks.ReentrantLock;
public class TryTest {
private final static ReentrantLock clientLock = new ReentrantLock();
public static final String test() {
String t = "";
try {
clientLock.lock();
t = "try";
System.out.println("加锁成功");
return t;
} catch (Exception e) {
// result = "catch";
t = "catch";
} finally {
t = "finally";
return t;
clientLock.unlock();
System.out.println("解锁成功");
return t;
//加上返回finnally
}
}
public static void main(String[] args) {
System.out.println(TryTest.test());
}
}
执行结果:
加锁成功
解锁成功
finally
第三种情况(try和finally里均有return)
public class TryTest {
private final static ReentrantLock clientLock = new ReentrantLock();
public static final String test() {
String t = "";
try {
clientLock.lock();
t = "try";
System.out.println("加锁成功");
return t;
} catch (Exception e) {
// result = "catch";
t = "catch";
return t;
} finally {
t = "finally";
clientLock.unlock();
System.out.println("解锁成功");
return t;
//加上返回finnally
}
}
public static void main(String[] args) {
System.out.println(TryTest.test());
}
}
执行结果:
加锁成功
解锁成功
try
结果分析
首先我们要知道,在try-catch-finally语句中,不管try语句块中code(代码)执行时是否抛出异常,finally语句是一定会执行的。那么问题来了,try{}中的return,finally{}中的code,finally里的return,他们三者的执行顺序究竟是怎么样的?
不难发现,这个问题其实在笔者上面列举的第三种情况的例子已经揭晓了答案。我们可以这样去记忆,首先try{}中的code先执行,当执行到return语句时,它和一般时候不一样,它并没有将值直接返回,而是存入了一个专门存return返回值的盒子中(我们可以这样去生活化的理解),盒子装入了try{}中的return返回值,然后接着再执行finally{}中的code,如果finally{}中也存在return,我们会将它也装入return盒子,而且覆盖掉之前存入的值,最后再返回此时return盒子中的值。
再细化的说就是,return中code的执行,和执行“返回”这个动作二者并不是同步的,“返回”这个动作是要等到执行完finally{}后才做的。