在JAVA面试中经常出现的题型
在学习try-catch语句块的时候,基本数据类型、引⽤类型、String在return finally的使⽤中要谨慎,注意输出台的输出结果,比较出各个数据类型的输出结果,就可以比较出这几个的区别了。观察以下几个代码运行进行了解其中的不同。
(1)第一个是基本数据类型,main方法中执行时调用了testReturn1()方法,在try中return返回基本数据类型i时,会先将返回值保存在一个地方(这一块我没有深究),此时保存的是一个int型的数(2),而不是一个变量,然后再去运行finally中的代码,那么此时finally中的变量i在做自加的时候,也不会影响已经生成的返回值。最终返回的是已经保存好的值。
public class ReturnaTest1 {
public static void main(String[] args) {
int i = new ReturnaTest1().testReturn1();
System.out.println(i);
}
private int testReturn1() {
int i = 1;
try {
i++;
System.out.println("try:" + i);
return i;
} finally {
i++;
System.out.println("finally:" + i);
}
}
}
(2)第二个是引用类型,在try{ }中return保存的是引用(也可以认为是一个地址),你可以通过引用去找到它的实际对象,所以在这个过程中别人完全可以改变里面的实际对象的值,等你再去通过引用查看的时候,里面的实际对象值有可能已经发生了变化。在下面这个例子中,main方法中执行时调用了testReturn2()方法,这个方法中,创建了一个List集合,list.add()添加了一个5并输出,return 返回的是一个引用,并保存在一个地方,再去执行finally中的代码,list.add()再将6添加进集合中,此时引用没有改变,但是里面是实际对象值发生了改变,所以再通过引用取到的值也发生了改变。最终控制台输出的结果如下。
import java.util.ArrayList;
import java.util.List;
public class ReturnaTest2 {
public static void main(String[] args) {
List list = new ReturnaTest2().testReturn2();
System.out.println(list);
}
private List testReturn2() {
List list=new ArrayList<>();
try {
list.add(5);
System.out.println("try:" + list);
return list;
} finally {
list.add(6);
System.out.println("finally:" + list);
}
}
}
(3)String 。 做之前需要了解:String属于Java中的字符串类型,也是一个引用类型,是一个比较特殊的类型,String和上面的引用类型存在不同的地方。JVM为了提升性能和减少内存开销,避免字符串的重复创建,其维护了一块特殊的内存空间,也就是字符串池。在创建字符串的时候,会先在字符串池中寻找是否已经存在,不存在的话在字符串池中创建,存在的话直接用就好了。
在这个例子中,return返回的虽然是引用类型的String,但是它比较特殊,返回的是字符串池中已经存在的或者刚刚创建的字符串,返回对象直接指向字符串池中的具体字符串(How are),待finally执行完之后再去返回。所以在finally执行的str+=" you";只是将变量str改变,对常量池中的How are字符串没有改变,对指向字符串池的返回值也不会改变。
public class ReturnaTest3 {
public static void main(String[] args) {
String s = new ReturnaTest3().testReturn3();
System.out.println(s);
}
private String testReturn3() {
String str="How";
try {
str+=" are";
System.out.println("try:" + str);
return str;
} finally {
str+=" you";
System.out.println("finally:" + str);
}
}
}