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

ops/2025/3/5 11:48:26/

本文已收录至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.ppmy.cn/ops/163289.html

相关文章

Zookeeper 及 基于ZooKeeper实现的分布式锁

1 ZooKeeper 1.1 ZooKeeper 介绍 ZooKeeper是一个开源的分布式协调服务&#xff0c;它的设计目标是将那些复杂且容易出错的分布式一致性服务封装起来&#xff0c;构成一个高效可靠的原语集&#xff0c;并以一系列简单易用的接口提供给用户使用。 原语&#xff1a;操作系统或…

IDEA中Git版本回退终极指南:Reset与Revert双方案详解

目录 前言一、版本回退前置知识二、Reset方案&#xff1a;整体改写历史1、IDEA图形化操作&#xff08;推荐&#xff09;1.1、查看提交历史1.2、选择目标版本1.3、选择回退模式1.3.1、Soft&#xff08;推荐&#xff09;1.3.2、Mixed1.3.3、Hard&#xff08;慎用&#xff09;1.3.…

Qt中如何从头到尾自定义设计一个标题栏

使用qt的widget自定义设计标题栏 头文件TitleBar.h #pragma once#include <QWidget> #include<QLabel> #include<QPushButton>enum ButtonType {MIN_BUTTON 0,MIN_MAX_BUTTON,ONLY_CLOSE_BUTTON };class TitleBar : public QWidget {Q_OBJECTpublic:Titl…

DiffBoost:通过文本引导的扩散模型增强医学图像分割

简介 本推文介绍了一篇来自美国西北大学&#xff0c;机器与混合智能实验室的论文《DiffBoost: Enhancing Medical Image Segmentation via Text-Guided Diffusion Model》&#xff0c;发表在医学图像处理领域的顶级期刊《IEEE Transactions on Medical Imaging》。论文中提出了…

MySQL 数据库安全配置最佳实践

文章目录 MySQL 数据库安全配置最佳实践账户与权限管理账户最小化原则权限最小化配置密码策略强化 认证与访问控制禁用匿名账户启用安全认证 网络安全防护访问源限制禁用远程root访问启用SSL加密 日志审计与监控全量审计配置二进制日志管理 服务端安全加固关键参数配置文件权限…

【大模型基础_毛玉仁】0.概述

更多内容&#xff1a;XiaoJ的知识星球 【大模型基础_毛玉仁】 系列文章参考 系列文章 【大模型基础_毛玉仁】0.概述 【大模型基础_毛玉仁】1.1 基于统计方法的语言模型 更新中。。。。。。 参考 书籍&#xff1a;大模型基础_完整版.pdf Github&#xff1a;https://github.co…

如何为JAR设置定时重启?

AI越来越火了&#xff0c;我们想要不被淘汰就得主动拥抱。推荐一个人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;最重要的屌图甚多&#xff0c;忍不住分享一下给大家。点击跳转到网站 前面我们说过了如何将jar交由Systemctl管理&#xff0c;下面我们…

图论part1|图论理论基础、深度搜索、广度搜索

图论理论基础 图的基本概念 节点多个节点的连线&#xff08;Node Edge&#xff09;种类&#xff1a;有向图&无向图度&#xff08;degree&#xff09;&#xff1a;有几条边连接该节点&#xff0c;该节点就有几度。 出度&#xff1a;从该节点出发的边的个数入度&#xff1a…