当前位置: 首页 > article >正文

京东一面:为什么 IDEA 建议去掉 StringBuilder,而要使用 “+” 拼接字符串?

本文已收录至Java面试网站:https://topjavaer.cn

今天咱们来聊聊一个很常见的开发场景:字符串拼接。在日常开发中,字符串拼接几乎是每个 Java 开发者都会用到的操作。但最近,有朋友在面试时被问到一个问题:“为什么 IDEA 建议用‘+’拼接字符串,而不是用 StringBuilder?”这问题听起来是不是有点反直觉?毕竟,在大家的普遍认知中,用 StringBuilder 拼接字符串效率更高。


一、“+” 拼接字符串

先来说说“+”拼接字符串。在 Java 中,“+” 是一个非常直观的字符串拼接操作符。比如,"Hello" + " " + "World",结果就是 "Hello World"。简单、直接、易读,这是它的优点。

但长期以来,我们一直被告知:“+”拼接字符串效率很低,尤其是在循环中。因为每次拼接都会创建一个新的字符串对象,导致大量的临时对象产生,增加了垃圾回收的负担。所以,很多开发者会习惯性地使用 StringBuilder 来代替“+”,尤其是在处理复杂的字符串拼接时。

然而,从 JDK 5 开始,Java 编译器做了一个优化——当你使用“+”拼接字符串时,编译器会自动将其优化为使用 StringBuilder 的方式。也就是说,"a" + "b" 在编译后,实际上会被编译器转换为 new StringBuilder().append("a").append("b").toString()。这样一来,“+”拼接字符串的性能问题就得到了解决。


二、实践验证

为了验证这一点,我们来做一个简单的实验。写一个测试类,分别用“+”和 StringBuilder 拼接字符串,然后比较它们的性能。

public String concatenationStringByPlus(String prefix, int i) {
    return prefix + "-" + i;
}

public String concatenationStringByStringBuilder(String prefix, int i) {
    return new StringBuilder().append(prefix).append("-").append(i).toString();
}

然后,我们用 JUnit 测试用例分别调用这两种方法,拼接 100000 次,看看耗时情况:

@Test
public void testStringConcatenationByPlus() {
    long startTime = System.currentTimeMillis();
    for (int i = 0; i < 100000; i++) {
        concatenationStringByPlus("testByPlus:", i);
    }
    long endTime = System.currentTimeMillis();
    System.out.println("使用 '+' 拼接 100000 次,耗时:" + (endTime - startTime) + " 毫秒");
}

@Test
public void testStringConcatenationByStringBuilder() {
    long startTime = System.currentTimeMillis();
    for (int i = 0; i < 100000; i++) {
        concatenationStringByStringBuilder("testByStringBuilder:", i);
    }
    long endTime = System.currentTimeMillis();
    System.out.println("使用 StringBuilder 拼接 100000 次,耗时:" + (endTime - startTime) + " 毫秒");
}

运行结果:

使用 '+' 拼接 100000 次,耗时:33 毫秒
使用 StringBuilder 拼接 100000 次,耗时:36 毫秒

可以看到,两者的耗时几乎一致。这说明在普通拼接场景下,“+” 和 StringBuilder 的性能几乎没有区别。而且,“+” 的代码更简洁、更易读


三、循环拼接的“陷阱”

那么,是不是在所有场景下,“+” 都和 StringBuilder 一样高效呢?答案是否定的。当涉及到循环拼接时,“+” 的效率问题就暴露出来了。

我们再做一个实验,模拟循环拼接一个长字符串。这次,我们分别用“+”和 StringBuilder 来拼接 10000 次:

@Test
public void testLoopConcatenationByPlus() {
    long startTime = System.currentTimeMillis();
    String str = "Initial String";
    for (int i = 0; i < 10000; i++) {
        str = str + "-" + i;
    }
    long endTime = System.currentTimeMillis();
    System.out.println("使用 '+' 循环拼接 10000 次,耗时:" + (endTime - startTime) + " 毫秒");
}

@Test
public void testLoopConcatenationByStringBuilder() {
    long startTime = System.currentTimeMillis();
    StringBuilder sb = new StringBuilder("Initial String");
    for (int i = 0; i < 10000; i++) {
        sb.append("-").append(i);
    }
    long endTime = System.currentTimeMillis();
    System.out.println("使用 StringBuilder 循环拼接 10000 次,耗时:" + (endTime - startTime) + " 毫秒");
}

运行结果:

使用 '+' 循环拼接 10000 次,耗时:463 毫秒
使用 StringBuilder 循环拼接 10000 次,耗时:13 毫秒

可以看到,循环拼接时,“+” 的效率远远低于 StringBuilder。这是因为每次循环时,“+” 都会创建一个新的 StringBuilder 对象,而 StringBuilder 只需要在同一个对象上追加内容,效率自然更高


四、为什么 IDEA 建议用“+”?

既然在循环拼接中 StringBuilder 更高效,为什么 IDEA 还要建议用“+”呢?原因在于编译器的优化和代码的可读性

对于简单的字符串拼接,编译器会自动将“+”优化为 StringBuilder 的形式。在这种情况下,使用“+”不仅代码更简洁,而且性能也一样好。而 StringBuilder 的代码相对冗长,可读性稍差

但如果是在循环中拼接字符串,IDEA 通常会提示你使用 StringBuilder,因为它能明显提升性能


五、总结

通过以上实验和分析,我们可以得出以下结论:

  1. 普通拼接场景:使用“+”和使用 StringBuilder 的性能几乎一致。由于“+”的代码更简洁、更易读,推荐在普通拼接场景下使用“+”。
  2. 循环拼接场景:推荐使用 StringBuilder。因为“+”在循环中会不断创建新的 StringBuilder 对象,导致性能大幅下降,而 StringBuilder 只需要初始化一次,效率更高。

最后分享一份大彬精心整理的大厂面试手册,包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等高频面试题,非常实用,有小伙伴靠着这份手册拿过字节offer~

需要的小伙伴可以自行下载

http://mp.weixin.qq.com/s?__biz=Mzg2OTY1NzY0MQ==&mid=2247485445&idx=1&sn=1c6e224b9bb3da457f5ee03894493dbc&chksm=ce98f543f9ef7c55325e3bf336607a370935a6c78dbb68cf86e59f5d68f4c51d175365a189f8#rd

围观朋友⭕:dabinjava


http://www.kler.cn/a/568389.html

相关文章:

  • 人工智能核心技术初识:Prompt、RAG、Agent与大模型微调
  • ECharts组件封装教程:Vue3中的实践与探索
  • Vue核心知识:KeepLive全方位分析
  • 从业务到技术:构建高效的 ERP 系统开发组织之道
  • deepseek使用记录18——文化基因之文化融合
  • 【CCF GESP 3 级】小猫分鱼 洛谷 B3925
  • 25年第二周:读书笔记
  • 【前端基础】1、HTML概述(HTML基本结构)
  • Rk3568驱动开发_点亮led灯代码完善(手动挡)_6
  • Windows权限维持之不死马(一)
  • Python 数据结构 4.单向链表
  • PHP:IDEA开发工具配置XDebug,断点调试
  • Android Coil3配置Application单例ImageLoader,Kotlin
  • upload
  • python流水线自动化项目教程
  • 从零开始开发纯血鸿蒙应用之语音朗读
  • Python核心技术,Django学习基础入门教程(附环境安装包)
  • 国自然面上项目|基于多模态MR影像的胶质母细胞瘤高危区域定位及预后预测研究|基金申请·25-02-28
  • LINUX基础 - 网络基础 [一]
  • 【ComfyUI】[进阶工作流] 高级采样器与Refiner的工作流优化