public static void test1(){
String str1="Hello world";
String str2=str1;
str1="Hello";
System.out.println(str2);
}
输出是Hello World
public static String getString(){
String s1="1";
try{
return s1;
}finally{
s1="2";
}
}
输出是1
static class Container{
String str="x";
}
public static Container getContainer(){
Container s1=new Container();
try{
s1.str="xx";
return s1;
}finally{
s1.str="yy";
}
}
输出是yy
初看到这个代码,可能有点疑惑。但仔细分析,关键点在于java中值传递和引用传递的深刻理解上。
对于方法来说,任何操作都是在自己的栈上进行的,因此不可能存放很大的值比如对象的值,只能存放一些小类型,也就是原始类型,int,boolean,char等。
所以对于大数据的对象都是在堆上面分配空间new出来,然后在栈上保存一个对该堆上地址的引用。
String类也不例外,因为每次“”都是一个new的过程,只是在String特有的char堆里面new出来而已。
基于上述理解,我们分析一下开头关于finally的困惑。
getString()方法返回的是“1”的堆地址,而相应的finally块中又new出了一个新的String对象“2”,两者互不干涉。
Container()方法返回的是new Container()的堆地址,而相应的fanally块改变的是该堆地址上该Container实例里面的属性值,所以是确确实实改变了的。