上一篇:大学生“暗网”发现“财富密码”,结果悲剧了~

作者:后青春期的Keats

来源:https://www.cnblogs.com/keatsCoder/

引言

都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差。最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的部分理解是错误的。

后来我通过实践测试 + 找原理 的方式搞清楚了这块的逻辑。现在将过程分享给大家

测试用例

我们的代码在循环中拼接字符串一般有两种情况

  • 第一种就是每次循环将对象中的几个字段拼接成一个新字段,再赋值给对象

  • 第二种操作是在循环外创建一个字符串对象,每次循环向该字符串拼接新的内容。循环结束后得到拼接好的字符串

对于这两种情况,我创建了两个对照组

第一组:

在每次 For 循环中拼接字符串,即拼即用、用完即毁。分别使用 String 和StringBuilder 拼接

/*** 循环内 String 拼接字符串,一次循环后销毁*/public static void useString(){for (int i = 0; i < CYCLE_NUM_BIGGER; i++) {String str = str1 + i + str2 + i + str3 + i + str4 ;}}/*** 循环内 使用 StringBuilder 拼接字符串,一次循环后销毁*/public static void useStringBuilder(){for (int i = 0; i < CYCLE_NUM_BIGGER; i++) {StringBuilder sb = new StringBuilder();String s = sb.append(str1).append(i).append(str2).append(i).append(str3).append(i).append(str4).toString();}}

第二组:

多次 For 循环拼接一个字符串,循环结束后使用字符串,使用后由垃圾回收器回收。也是分别使用 String 和 StringBuilder 拼接

/*** 多次循环拼接成一个字符串 用 String*/public static void useStringSpliceOneStr (){String str = "";for (int i = 0; i < CYCLE_NUM_LOWER; i++) {str += str1 + str2 + str3 + str4 + i;}}/*** 多次循环拼接成一个字符串 用 StringBuilder*/public static void useStringBuilderSpliceOneStr(){StringBuilder sb = new StringBuilder();for (int i = 0; i < CYCLE_NUM_LOWER; i++) {sb.append(str1).append(str2).append(str3).append(str4).append(i);}}

为了保证测试质量,在每个测试项目进行前。线程休息 2s,之后空跑 5 次热身。最后执行 5 次求平均时间的方式计算时间

public static int executeSometime(int kind, int num) throws InterruptedException {Thread.sleep(2000);int sum = 0;for (int i = 0; i < num + 5; i++) {long begin = System.currentTimeMillis();switch (kind){case 1:useString();break;case 2:useStringBuilder();break;case 3:useStringSpliceOneStr();break;case 4:useStringBuilderSpliceOneStr();break;default:return 0;}long end = System.currentTimeMillis();if(i > 5){sum += (end - begin);}}return sum / num;}

主方法

public class StringTest {public static final int CYCLE_NUM_BIGGER = 10_000_000;public static final int CYCLE_NUM_LOWER = 10_000;public static final String str1 = "张三";public static final String str2 = "李四";public static final String str3 = "王五";public static final String str4 = "赵六";public static void main(String[] args) throws InterruptedException {int time = 0;int num = 5;time = executeSometime(1, num);System.out.println("String拼接 "+ CYCLE_NUM_BIGGER +" 次," + num + "次平均时间:" + time + " ms");time = executeSometime(2, num);System.out.println("StringBuilder拼接 "+ CYCLE_NUM_BIGGER +" 次," + num + "次平均时间:" + time + " ms");time = executeSometime(3, num);System.out.println("String拼接单个字符串 "+ CYCLE_NUM_LOWER +" 次," + num + "次平均时间:" + time + " ms");time = executeSometime(4, num);System.out.println("StringBuilder拼接单个字符串 "+ CYCLE_NUM_LOWER +" 次," + num + "次平均时间:" + time + " ms");}
}

测试结果

测试结果如下

结果分析

第一组

10_000_000 次循环拼接,在循环内使用 String 和 StringBuilder 的效率是一样的!为什么呢?

使用 javap -c StringTest.class 反编译查看两个方法编译后的文件:

可以发现 String 方法拼接字符串编译器优化后使用的就是 StringBuilder、因此用例1 和用例2 的效率是一样的。

第二组

第二组的结果就是大家喜闻乐见的了,由于 10_000_000 次循环String 拼接实在太慢所以我采用了 10_000 次拼接来分析。

分析用例3:虽然编译器会对 String 拼接做优化,但是它每次在循环内创建 StringBuilder 对象,在循环内销毁。下次循环他有创建。相比较用例4在循环外创建,多了 n 次 new 对象、销毁对象的操作、n - 1 次将 StringBuilder 转换成 String 的操作 。效率低也是理所应当了。

扩展

第一组的测试还有一种写法:

/*** 循环内 使用 StringBuilder 拼接字符串,一次循环后销毁*/public static void useStringBuilderOut(){StringBuilder sb = new StringBuilder();for (int i = 0; i < CYCLE_NUM_BIGGER; i++) {
//            sb.setLength(0);sb.delete(0, sb.length());String s = sb.append(str1).append(i).append(str2).append(i).append(str3).append(i).append(str4).toString();}}

循环外创建 StringBuilder 每次循环开始的时候清空 StringBuilder 的内容然后拼接。这种写法无论使用 sb.setLength(0); 还是 sb.delete(0, sb.length()); 效率都比直接在循环内使用 String / StringBuilder 慢。奈何才疏学浅我一直想不明白为什么他慢。我猜测是 new 对象的速度比重置长度慢,于是这样测试了以下:

public static void createStringBuider() {for (int i = 0; i < CYCLE_NUM_BIGGER; i++) {StringBuilder sb = new StringBuilder();}}public static void cleanStringBuider() {StringBuilder sb = new StringBuilder();for (int i = 0; i < CYCLE_NUM_BIGGER; i++) {sb.delete(0, sb.length());}}

但是结果是 cleanStringBuider 更快。让我摸不着头脑。

如果有大神看到希望可以帮忙分析分析

结论

  • 编译器会将 String 拼接优化成使用 StringBuilder,但是还是有一些缺陷的。主要体现在循环内使用字符串拼接,编译器不会创建单个 StringBuilder 以复用

  • 对于多次循环内拼接一个字符串的需求:StringBuilder 很快,因为其避免了 n 次 new 对象、销毁对象的操作,n - 1 次将 StringBuilder 转换成 String 的操作

  • StringBuilder 拼接不适用于循环内每次拼接即用的操作方式。因为编译器优化后的 String 拼接也是使用 StringBuilder 两者的效率一样。后者写起来还方便...

感谢您的阅读,也欢迎您发表关于这篇文章的任何建议,关注我,技术不迷茫!小编到你上高速。

· END ·

最后,关注公众号互联网架构师,在后台回复:2T,可以获取我整理的 Java 系列面试题和答案,非常齐全。

正文结束

推荐阅读 ↓↓↓

1.不认命,从10年流水线工人,到谷歌上班的程序媛,一位湖南妹子的励志故事

2.如何才能成为优秀的架构师?

3.从零开始搭建创业公司后台技术栈

4.程序员一般可以从什么平台接私活?

5.37岁程序员被裁,120天没找到工作,无奈去小公司,结果懵了...

6.IntelliJ IDEA 2019.3 首个最新访问版本发布,新特性抢先看

7.这封“领导痛批95后下属”的邮件,句句扎心!

8.15张图看懂瞎忙和高效的区别!

一个人学习、工作很迷茫?

点击「阅读原文」加入我们的小圈子!

在 for 循环里写加号拼接字符串的那个同事,不见了~相关推荐

  1. 昨天还在 for 循环里写加号拼接字符串的那个同事,今天已经不在了

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 来源: 测试用例 测试结果 结果分析 第一组 第二组 扩展 结论 ...

  2. 昨天还在for循环里写加号拼接字符串的那个同事,今天已经不在了

    点击上方 好好学java ,选择 星标 公众号重磅资讯,干货,第一时间送达 今日推荐:分享一套基于SpringBoot和Vue的企业级中后台开源项目,这个项目有点哇塞!个人原创100W +访问量博客: ...

  3. 小程序循环里做字符串拼接_昨天还在for循环里写加号拼接字符串的那个同事,今天已经不在了...

    引言 都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差.最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的 ...

  4. for循环优化_昨天还在for循环里写加号拼接字符串的那个同事,今天已经不在了

    引言 都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差.最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的 ...

  5. 还在 for 循环里使用 + 拼接字符串?明天不用来上班了!

    欢迎关注方志朋的博客,回复"666"获面试宝典 来源:cnblogs.com/keatsCoder/p/13212289.html 引言 都说 StringBuilder 在处理字 ...

  6. 你还在 for 循环里使用 + 拼接字符串吗?

    引言 都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差.最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的 ...

  7. java 拼接字符串的方法

    1.拼接字符串的方法,先要将字符串转化为数字类型,再根据需要拼接.这样可以避免直接拼接导致的错误. 2.将字符串转化为数字类型,这个就是一个循环.可以使用循环的方法,但是循环次数不宜太多,否则容易出错 ...

  8. shell for循环拼接字符串

    # 在当前目录下创建文本文件temp,如果文件存在则清空文件 $(> temp) # for 循环将参数追加到当前目录的temp文件,逗号分隔,echo -n 不换行 for i in $*;d ...

  9. 【C 语言】文件操作 ( 配置文件读写 | 写出或更新配置文件 | 逐行遍历文件文本数据 | 获取文件中的文本行 | 查询文本行数据 | 追加文件数据 | 使用占位符方式拼接字符串 )

    文章目录 一.逐行遍历文件文本数据 1.获取文件中的文本行 2.查询文本行数据 3.追加文件数据 4.使用占位符方式拼接字符串 二.完整代码示例 一.逐行遍历文件文本数据 1.获取文件中的文本行 调用 ...

  10. python里的拼接_Python拼接字符串的7种方法总结

    前言 忘了在哪看到一位编程大牛调侃,他说程序员每天就做两件事,其中之一就是处理字符串.相信不少同学会有同感. 在Python中,我们经常会遇到字符串的拼接问题,几乎任何一种编程语言,都把字符串列为最基 ...

最新文章

  1. Google的预训练模型又霸榜了,这次叫做T5(附榜单)
  2. 支付宝支付 第九集:产品数据和支付二维码对接
  3. 大数据WEB工具Hue
  4. 禁止鼠标右键的代码(转)
  5. 【错误记录】Git 使用报错 ( git branch -a 仍能查询到已经删除的远程分支 )
  6. 计算硼原子的基态能级B---交换能
  7. MATLAB实战系列(二十六)-matlab通过遗传算法求解车间调度问题
  8. 超详细图解!【MySQL进阶篇】存储过程,视图,索引,函数,触发器
  9. 【手把手 脑把脑】教会你使用idea基于MapReduce的统计数据分析(从问题分析到代码编写)
  10. Spark内核解析1
  11. Android 系统(224)---如何不显示开机SIM卡欢迎语
  12. 可以分屏的软件_Screen分屏软件下载|Screen+专业分屏 免费版v1.4.25 下载
  13. EF框架step by step(1)—Database-First
  14. 数字化转型中平台思维的十大要素-《数字化转型的道与术》
  15. java 软键盘_【学习笔记】【java appium】软键盘搜索、回车按钮
  16. html仿微信评论输入框,简单仿微信朋友圈评论功能
  17. 搭建视频会议系统OpenMeetings
  18. PDF删除页面技巧介绍
  19. 互联网产品推广分为哪些阶段?
  20. C++中的 .h 和 .cpp 详解

热门文章

  1. EasyRecovery,重新找寻丢失的文件
  2. 音乐后期处理:音乐失真效果制作
  3. mac下selenium+python环境搭建
  4. web.config中的ExtensionlessUrlHandler-Integrated-4.0
  5. java中Action层、Service层和Dao层的功能区分
  6. 记一次失败的电话面试
  7. 微信全球MBA创新大赛Roadshow最终站火爆中欧
  8. System Center 2012 R2 支持的SQL版本
  9. 简单总结.NET中的各% %用法及区别
  10. xcode快捷键(二)