一道笔试题来理顺Java中的值传递和引用传递

前段时间参加了一场面试,其中有一道引用传递的题,由于当时并无考虑清楚因此作错了。 
如今来复盘一下,题目以下:
html

private static void change(StringBuffer str11, StringBuffer str12) {

str12 = str11;
str11 = new StringBuffer("new world");
str12.append("new world");
}

public static void main(String[] args) {
StringBuffer str1 = new StringBuffer("good ");
StringBuffer str2 = new StringBuffer("bad ");
change(str1, str2);
System.out.println(str1.toString());
System.out.println(str2.toString());
}
 
就是这么一个很简单的题目,你们应该知道答案吧? 答案是:

good new world 
badweb

下面就来复盘一下为什么是这两个答案,下面会作详细的图文分析。 
要搞明白这个问题,那么就须要明白Java中的值传递和引用传递了。
面试

Java对象参数传递虽然传递的是地址(引用),但仍然是值调用。是时候须要给引用调用和值调用一个准确的定义了。编程

  • 值调用(call by value): 在参数传递过程当中,形参和实参占用了两个彻底不一样的内存空间。形参所存储的内容是实参存储内容的一份拷贝。实际上,Java对象的传递就符合这个定义,只不过形参和实参所储存的内容并非常规意义上的变量值,而是变量的地址。咳,回过头想一想:变量的地址不也是一种值吗!app

  • 引用调用(call by reference) : 在参数传递的过程当中,形参和实参彻底是同一块内存空间,二者不分彼此。 实际上,形参名和实参名只是编程中的不一样符号,在程序运行过程当中,内存中存储的空间才是最重要的。不一样的变量名并不能说明占用的内存存储空间不一样。ide

那么接下来就画图分析文章开头抛出的问题: 
这里再把代码贴一遍,你们能够对比着代码和图一块儿看。
函数

private static void change(StringBuffer str11, StringBuffer str12) {

str12 = str11;//2
str11 = new StringBuffer("new world");//3
str12.append("new world");//4
}//5

public static void main(String[] args) {
StringBuffer str1 = new StringBuffer("good ");
StringBuffer str2 = new StringBuffer("bad ");
change(str1, str2);//1
System.out.println(str1.toString());
System.out.println(str2.toString());
}
 
  • 执行到第一步的图: 
    spa

     

  • 执行到第二步的图,这里chage方法中的形参str11,str22是实参str1,str2的地址拷贝。 
    orm

     

  • 这里str11和str22是实参str1,str2的地址拷贝,接着执行第三步的图: 
    htm

     

  • 上图中str12地址指向了“good”,接着看第四步图: 

     

  • 上图中str11的地址指向了堆中新的对象"new world",接着看第五步的图: 

     

这里图算是画完了,确实不太擅长画图,你们就先勉强看吧。 
这里有一个很重要的点就是StringBuffer是可变的,具体解释你们能够搜索下。

由于这里str12.append("new world")直接修改了good的值。因此main函数中打印的会是:good new world。

到这里就算讲完了,一个很简单的例子,这里算是对以前一道笔试题的复盘,有不对的地方欢迎你们指正。

相关文章
相关标签/搜索