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

前段时间参加了一场面试,其中有一道引用传递的题,因为当时并没有考虑清楚所以做错了。 现在来复盘一下,题目如下:

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 bad

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

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

  • 值调用(call by value): 在参数传递过程中,形参和实参占用了两个完全不同的内存空间。形参所存储的内容是实参存储内容的一份拷贝。实际上,Java对象的传递就符合这个定义,只不过形参和实参所储存的内容并不是常规意义上的变量值,而是变量的地址。咳,回过头想想:变量的地址不也是一种值吗!
  • 引用调用(call by reference) : 在参数传递的过程中,形参和实参完全是同一块内存空间,两者不分彼此。 实际上,形参名和实参名只是编程中的不同符号,在程序运行过程中,内存中存储的空间才是最重要的。不同的变量名并不能说明占用的内存存储空间不同。

那么接下来就画图分析文章开头抛出的问题: 这里再把代码贴一遍,大家可以对比着代码和图一起看。

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());
}
  • 执行到第一步的图:
  • 执行到第二步的图,这里chage方法中的形参str11,str22是实参str1,str2的地址拷贝。
  • 这里str11和str22是实参str1,str2的地址拷贝,接着执行第三步的图:
  • 上图中str12地址指向了“good”,接着看第四步图:
  • 上图中str11的地址指向了堆中新的对象"new world",接着看第五步的图:

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

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

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

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏机器学习从入门到成神

字符串面试题(二)— 间隔字符串逆序

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/sinat_35512245/articl...

853
来自专栏androidBlog

归并排序 递归版和非递归版的实现(java)

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/gdutxiaoxu/article/details/...

1141
来自专栏决胜机器学习

PHP数据结构(三)——运用栈实现括号匹配

PHP数据结构(三)——运用栈实现括号匹配 (原创内容,转载请注明来源,谢谢) 栈在数据结构上是一种特殊的线性表,其限制是仅允许在表的一端进行插入和删除运算,...

3796
来自专栏水击三千

浅谈JavaScript的面向对象程序设计(四)

  本文继续讲解JavaScript的面向对象程序设计。继承是面向对象语言中的一个基本概念,面向对象语言支持两种继承实现方式:接口继承和实现继承。接口继承只继承...

2719
来自专栏GreenLeaves

C# 终极基类Object介绍

一、简介 Object这个类型,相信everyone都不陌生,这个是CLR定义的最基础的类型,俗称"上帝类"。CLR(运行时)要求所有类型,不管是系统定义的类型...

1786
来自专栏大前端开发

ES6特性之:箭头函数

在ES6的所有新特性中,箭头函数(Arrow Fucntion)算是我用的最频繁的特性之一了。

781
来自专栏企鹅号快讯

Java字符串的10大热点问题盘点

往期精选 下面我为大家总结了10条Java开发者经常会提的关于Java字符串的问题,如果你也是Java初学者,仔细看看吧: 1、如何比较字符串,应该用”==”还...

3418
来自专栏java学习

8张图理解Java

1、字符串不变性 下面这张图展示了这段代码做了什么 1、String s = "abcd"; 2、s = s.concat("ef"); ? 2、equals(...

27310
来自专栏hbbliyong

C++虚析构函数解析

 当派生类对象从内存中撤销时一般先运行派生类的析构函数,然后再调用基类的析构函数。 如果用new运算符建立的派生类的临时对象,对指向基类的指针指向这个临时对...

3367
来自专栏编程

Python的类和对象

对象=属性(特征)+方法(行为) 类:在python中,把具有相同属性和方法的对象归为一个类(class) self: init()构造方法,只要实例化一个对象...

17710

扫码关注云+社区