老读者都知道了,六年前,我从苏州回到洛阳,抱着一幅“海归”的心态,投了不少简历,也“约谈”了不少面试官,但仅有两三个令我感到满意。其中有一位叫老马,至今还活在我的手机通讯录里。他当时扔了一个面试题把我砸懵了:“王二,Java 字符串可以引用传递吗?”

当时二十三岁,正值青春年华,从事 Java 编程已有 N 年经验(N < 4),自认为所有的面试题都能对答如流,结果没想到啊,被“刁难”了——原来洛阳这块互联网的荒漠也有技术专家啊。现在回想起来,脸上不自觉地泛起了羞愧的红晕:主要是自己当时太菜了。不管怎么说,是时候写篇文章剖析一下字符串是否可以引用传递了。

对于绝大多数的初级程序员或者说不重视“内功”的老鸟来说,往往停留在“知其然不知其所以然”的层面上——会用,略知一二,但要求他把问题说清楚的时候,就只能挠挠头双手一摊一张问号脸了。


好了,让我们来步入正题。先来看一段有趣但令人困惑的代码片段吧。

public static void main(String[] args) {
    String x = new String("沉默王二");
    change(x);
    System.out.println(x);
}

public static void change(String x) {
    x = "沉默王三";
}

从代码的字面逻辑来看,程序应该输出“沉默王三”,但事与愿违,程序输出的结果却是“沉默王二”。change() 方法做的是无用功,因为 String 是值传递而不是引用传递。引用传递可以在被调用的方法中对实参进行修改,但值传递却不可以。为什么呢?

x 存储的是一个引用,该引用指向内存中的“沉默王二”字符串对象。当我们把 x 作为参数传递给 change() 方法时,x 仍然指向的是内存中“沉默王二”字符串,就像下面这幅图表达的意思一样。


那么问题来了。正因为 Java 是值传递,x 的值是“沉默王二”的引用。那么当 change() 方法被调用的时候,x 不是刚好指向了内存中新创建的字符串对象“沉默王三”了吗?就像下面这幅图表达的意思那样。


哦,看起来是一个很完美的解释,对吧?但这样的解释存在一些问题。

当字符串“沉默王二”被创建的时候,Java 会在内存中申请一小段空间,用来存储这个字符串对象。然后呢,把对象的引用指向了变量 x,也就是说,变量 x 实际上存储的是对象的引用(对象在内存中存储的地址)。

我相信大家对上面这一点(对象和对象引用)已经完全理解了。

关键的点来了。当变量 x 作为参数(实参)传递给 change() 方法时,实际上传递的是 x 的一个拷贝(形参)。在 change() 方法中,形参 x 起先引用的也是“沉默王二”这个对象,当执行 x = "沉默王三" 的时候,会在内存中创建新的字符串“沉默王三”,然后形参 x 不再引用“沉默王二”这个对象了,改为引用“沉默王三”这个对象了。但实参 x 呢?并没有发生任何的改变!就像下面这幅图一样。


假如我们真的需要改变字符串呢?那就不能使用 String 类了,最好使用 StringBuilder,来撸一串代码吧。

public static void main(String[] args) {
    StringBuilder x = new StringBuilder("沉默王二");
    change(x);
    System.out.println(x);
}

public static void change(StringBuilder x) {
    x.delete(3,4).append("三");
}

上述代码会输出“沉默王三”,但假如我们使用 new 关键字重新对形参 x 进行赋值,就无济于事。

public static void main(String[] args) {
    StringBuilder x = new StringBuilder("沉默王二");
    change(x);
    System.out.println(x);
}

public static void change(StringBuilder x) {
    x = new StringBuilder("沉默王三");
}

程序输出的结果仍然是“沉默王二”,原因其实和 String 一样,change() 方法在内存中创建了新的字符串“沉默王三”,然后形参 x 不再引用“沉默王二”这个对象,改为引用“沉默王三”这个对象了。但实参 x 并没有任何改变。

看到这,有些读者可能更疑惑了。x = new StringBuilder("沉默王三") 不可以改变实参,而 x.delete(3,4).append("三") 却可以,为什么?为什么?为什么?为什么呢?

不要着急,我们来分析一下 delete() 方法的源码。

public AbstractStringBuilder delete(int start, int end) {
    int len = end - start;
    if (len > 0) {
        System.arraycopy(value, start+len, value, start, count-end);
        count -= len;
    }
    return this;
}

其中 value 是一个字符数组,用来存储字符序列;count 用来表示字符序列中实际有效的字符数量。

count -= len 执行之前,value 的字符内容为“沉默王二”,count 为 4。我是怎么知道的呢?通过 IDEA 的 debug 视图,截图为证。


count -= len 执行之后,value 的字符内容仍然为“沉默王二”,但 count 变成了 3。


当鼠标停留在 this 上时,此时的字符内容为“沉默王”,也就意味着 x 当前的字符内容为“沉默王”。同样的,当我们在 append() 方法上进行 debug 的时候,也可以观察到字符串发生变化的细节。


append() 方法执行结束后,此时形参 x 的字符内容为“沉默王三”。


change() 方法执行完后,此时实参 x 的字符内容为“沉默王三”。


通过上面的源码分析,大家应该会发现另外一个事实:x 对象始终是“StringBuilder@512”,这意味着什么呢?一图胜千言,画个图大家一看就明白了。


由于形参 x 和实参 x 引用的都是同一个对象,那么 change() 方法执行结束后,实参 x 的字符内容自然也就发生了变化。

综上所述:Java 字符串不是引用传递而是值传递;更进一步的说,Java 只有值传递,没有引用传递。


遥想公瑾当年,小乔初嫁了,雄姿英发。

羽扇纶巾,谈笑间,樯橹灰飞烟灭。

故国神游,多情应笑我,早生华发。

哎,后悔啊,早年我要是能把这道面试题吃透的话,也不用被老马刁难了。另外,我想要告诉大家的是,作为程序员,我们千万不要轻视这些基础的知识点。因为基础的知识点是各种上层技术共同的基础,只有彻底地掌握了这些基础知识点,才能更好地理解程序的运行原理,做出更优化的产品。


好了,各位读者朋友们,以上就是本文的全部内容了。能看到这里的都是最优秀的程序员,升职加薪就是你了

面试官刁难:Java字符串可以引用传递吗?的更多相关文章

  1. java到底是引用传递还是值传递?

    今天我们来讲讲一个在学习中容易误解的问题,面试中也偶尔问到,java方法调用时到底是值传递还是引用传递? 首先,请大家来做一个判断题,下面的3个问题是否描述正确 1. java基本数据类型传递是值传递 ...

  2. Java中没有引用传递只有值传递(在函数中)

    ◆传参的问题 引用类型(在函数调用中)的传参问题,是一个相当扯的问题.有些书上说是传值,有些书上说是传引用.搞得Java程序员都快成神经分裂了.所以,我们最后来谈一下“引用类型参数传递”的问题. 如下 ...

  3. 面试官:Java中对象都存放在堆中吗?你知道逃逸分析?

    面试官:Java虚拟机的内存分为哪几个区域? 我(微笑着):程序计数器.虚拟机栈.本地方法栈.堆.方法区 面试官:对象一般存放在哪个区域? 我:堆. 面试官:对象都存放在堆中吗? 我:是的. 面试官: ...

  4. Java中的引用传递和值传递

    Java中的引用传递和值传递 关于Java的引用传递和值传递,在听了老师讲解后,还是没有弄清楚是怎么一回事,于是查了资料,所以在这里与大家分享,有不对的地方,欢迎大家留言. java中是没有指针的,j ...

  5. 死磕面试系列,Java到底是值传递还是引用传递?

    Java到底是值传递还是引用传递? 这虽然是一个老生常谈的问题,但是对于没有深入研究过这块,或者Java基础不牢的同学,还是很难回答得让人满意. 可能很多同学能够很轻松的背出JVM.分布式事务.高并发 ...

  6. 【性能优化】面试官:Java中的对象都是在堆上分配的吗?

    写在前面 从开始学习Java的时候,我们就接触了这样一种观点:Java中的对象是在堆上创建的,对象的引用是放在栈里的,那这个观点就真的是正确的吗?如果是正确的,那么,面试官为啥会问:"Jav ...

  7. 【对线面试官】Java NIO

    服务端: public class NoBlockServer { public static void main(String[] args) throws IOException { // 1.获 ...

  8. 面试官:Java从编译到执行,发生了什么?

    面试官:今天从基础先问起吧,你是怎么理解Java是一门「跨平台」的语言,也就是「一次编译,到处运行的」? 候选者:很好理解啊,因为我们有JVM. 候选者:Java源代码会被编译为class文件,cla ...

  9. 理解Java中的引用传递和值传递

    关于Java传参时是引用传递还是值传递,一直是一个讨论比较多的话题,有论坛说Java中只有值传递,也有些地方说引用传递和值传递都存在,比较容易让人迷惑.关于值传递和引用传递其实需要分情况看待,今天学习 ...

随机推荐

  1. iOS 9 适配系列教程

    http://www.cocoachina.com/ios/20150703/12392.html 本文是投稿文章,作者:ChenYilong(https://github.com/ChenYilon ...

  2. laravel重定向到上一个页面怎么带参数返回 withsucess 成功提示信息

    //控制器中 return back()->with('success','操作成功'); //with的参数1是一个session变量名,参数2为该session变量值,在视图直接这样获取 @ ...

  3. 15-8 pymysql的使用

    一 安装pymysql模块 1 pycharm安装 file-setting如图:然后点加号,搜索pymsql即可,点击安装 2 pip 安装 pip3 install pymysql 二  连接数据 ...

  4. LocalDate、LocalDateTime与timestamp、Date的转换

    LocalDate.LocalDateTime与timestamp.Date的转换 1.LocalDate转Date LocalDate nowLocalDate = LocalDate.now(); ...

  5. 多版本python共存,安装三方库到指定python版本

    多版本python安装过程略过不提提供完美解决python启动和各版本pip问题: python3下pip安装命令如下: py -3 -m pip install xxxxxx python2下pip ...

  6. iPython的安装过程

    http://blog.csdn.net/u012587561/article/details/50900781 python2.7.10 amd64 win10 x64 1. 安装setuptool ...

  7. 2019-8-31-PowerShell-通过-WMI-获取系统服务

    title author date CreateTime categories PowerShell 通过 WMI 获取系统服务 lindexi 2019-08-31 16:55:58 +0800 2 ...

  8. SuperSocket 中的日志系统

    当 SuperSocket boostrap 启动时,日志系统将会自动启动. 所以你无须创建自己的日志工具,最好直接使用SuperSocket内置的日志功能. SuperSocket 默认使用log4 ...

  9. Python--day72--SweetAlert插件

    引用:http://www.cnblogs.com/liwenzhou/p/8718861.html 补充一个SweetAlert插件示例 点击下载Bootstrap-sweetalert项目. $( ...

  10. Python--day64--找到作者关联的所有书籍对象、ORM多对多关联查询的原理

    找到当前作者关联的所有书籍对象: ORM多对多关联查询的原理: 编辑作者: