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

今天咱们来聊聊一个很常见的开发场景:字符串拼接。在日常开发中,字符串拼接几乎是每个 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 只需要初始化一次,效率更高。

围观朋友⭕:dabinjava

#java#
全部评论
那用Junit测试的使用+拼接字符串的时候,不会每次都新建string builder吗?
点赞 回复 分享
发布于 今天 12:55 吉林
学到了,受益嘞
点赞 回复 分享
发布于 今天 13:47 浙江
省流:拼一次直接+,大量循环拼建议显式使用StringBuilder拼接避免StringBuilder对象被多次创建
点赞 回复 分享
发布于 今天 14:49 江苏

相关推荐

01-14 12:08
门头沟学院 Java
神哥了不得:(非引流)1.既然发出来了简历,就稍微提一点点小建议,确实简历很不错了,练手项目可以换一些质量高的,工作内容,可以加上一些量化指标,比如第一条系统响应速度由多少变成多少,减少了百分之多少,第4条就很不错。2.广投,年前实习招募比较少了
点赞 评论 收藏
分享
02-22 20:28
重庆大学 Java
程序员牛肉:首先不要焦虑,你肯定是有希望的。 首先我觉得你得好好想一想自己想要什么。找不到开发岗就一定是失败的吗?那开发岗的35岁危机怎么说?因此无论是找工作还是考公我觉得你都需要慎重的想一想。但你一定要避开这样一个误区:“我是因为找不到工作所以不得不选择考公”。 千万不要这么想。你这个学历挺好的了,因此你投后端岗肯定是有面试机会的。有多少人简历写的再牛逼,直接连机筛简历都过不去有啥用?因此你先保持自信一点。 以你现在的水平的话,其实如果想要找到暑期实习就两个月:一个月做项目+深挖,并且不断的背八股。只要自己辛苦一点,五月份之前肯定是可以找到暑期实习的,你有点太过于高看大家之间的技术差距了。不要焦虑不要焦虑。 除此之外说回你这个简历内容的话,基本可以全丢了。如果想做后端,先踏踏实实做两个项目再说+背八股再说。如果想考公,那就直接备战考公。 但是但是就像我前面说的:你考公的理由可以是因为想追求稳定,想追求轻松。但唯独不能是因为觉得自己找不到工作。不能这么小瞧自己和自己的学历。
点赞 评论 收藏
分享
评论
2
1
分享

创作者周榜

更多
牛客网
牛客企业服务