在 for 循环里写加号拼接字符串的那个同事,不见了~
上一篇:大学生“暗网”发现“财富密码”,结果悲剧了~
作者:后青春期的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 循环里写加号拼接字符串的那个同事,不见了~相关推荐
- 昨天还在 for 循环里写加号拼接字符串的那个同事,今天已经不在了
点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 来源: 测试用例 测试结果 结果分析 第一组 第二组 扩展 结论 ...
- 昨天还在for循环里写加号拼接字符串的那个同事,今天已经不在了
点击上方 好好学java ,选择 星标 公众号重磅资讯,干货,第一时间送达 今日推荐:分享一套基于SpringBoot和Vue的企业级中后台开源项目,这个项目有点哇塞!个人原创100W +访问量博客: ...
- 小程序循环里做字符串拼接_昨天还在for循环里写加号拼接字符串的那个同事,今天已经不在了...
引言 都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差.最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的 ...
- for循环优化_昨天还在for循环里写加号拼接字符串的那个同事,今天已经不在了
引言 都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差.最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的 ...
- 还在 for 循环里使用 + 拼接字符串?明天不用来上班了!
欢迎关注方志朋的博客,回复"666"获面试宝典 来源:cnblogs.com/keatsCoder/p/13212289.html 引言 都说 StringBuilder 在处理字 ...
- 你还在 for 循环里使用 + 拼接字符串吗?
引言 都说 StringBuilder 在处理字符串拼接上效率要强于 String,但有时候我们的理解可能会存在一定的偏差.最近我在测试数据导入效率的时候就发现我以前对 StringBuilder 的 ...
- java 拼接字符串的方法
1.拼接字符串的方法,先要将字符串转化为数字类型,再根据需要拼接.这样可以避免直接拼接导致的错误. 2.将字符串转化为数字类型,这个就是一个循环.可以使用循环的方法,但是循环次数不宜太多,否则容易出错 ...
- shell for循环拼接字符串
# 在当前目录下创建文本文件temp,如果文件存在则清空文件 $(> temp) # for 循环将参数追加到当前目录的temp文件,逗号分隔,echo -n 不换行 for i in $*;d ...
- 【C 语言】文件操作 ( 配置文件读写 | 写出或更新配置文件 | 逐行遍历文件文本数据 | 获取文件中的文本行 | 查询文本行数据 | 追加文件数据 | 使用占位符方式拼接字符串 )
文章目录 一.逐行遍历文件文本数据 1.获取文件中的文本行 2.查询文本行数据 3.追加文件数据 4.使用占位符方式拼接字符串 二.完整代码示例 一.逐行遍历文件文本数据 1.获取文件中的文本行 调用 ...
- python里的拼接_Python拼接字符串的7种方法总结
前言 忘了在哪看到一位编程大牛调侃,他说程序员每天就做两件事,其中之一就是处理字符串.相信不少同学会有同感. 在Python中,我们经常会遇到字符串的拼接问题,几乎任何一种编程语言,都把字符串列为最基 ...
最新文章
- Google的预训练模型又霸榜了,这次叫做T5(附榜单)
- 支付宝支付 第九集:产品数据和支付二维码对接
- 大数据WEB工具Hue
- 禁止鼠标右键的代码(转)
- 【错误记录】Git 使用报错 ( git branch -a 仍能查询到已经删除的远程分支 )
- 计算硼原子的基态能级B---交换能
- MATLAB实战系列(二十六)-matlab通过遗传算法求解车间调度问题
- 超详细图解!【MySQL进阶篇】存储过程,视图,索引,函数,触发器
- 【手把手 脑把脑】教会你使用idea基于MapReduce的统计数据分析(从问题分析到代码编写)
- Spark内核解析1
- Android 系统(224)---如何不显示开机SIM卡欢迎语
- 可以分屏的软件_Screen分屏软件下载|Screen+专业分屏 免费版v1.4.25 下载
- EF框架step by step(1)—Database-First
- 数字化转型中平台思维的十大要素-《数字化转型的道与术》
- java 软键盘_【学习笔记】【java appium】软键盘搜索、回车按钮
- html仿微信评论输入框,简单仿微信朋友圈评论功能
- 搭建视频会议系统OpenMeetings
- PDF删除页面技巧介绍
- 互联网产品推广分为哪些阶段?
- C++中的 .h 和 .cpp 详解