Java又被喷了 String的append

赵姐夫在info q 上的ppt

开吵的缘由

姐夫说

看到群里有人贴这个。Java编译器真的会把普通的字符串拼接操作“优化”为StringBuilder?有这么糟蹋性能的做法吗? ​​​​

嗯... 小萌新表示 还好吧. 否则每次在常量空间中创建一个中间值 这样子多不值当. (虽然的在新的JVM标准中 已经把String 从常量空间挪到新生区域了

这是个严重影响性能的做法不是吗?明明目标字符串长度也可以知道,你搞个SB需要额外字符数组翻几次倍最后再复制一份,怪不得总是要和GC作斗争,这么个基础操作都这么浪费。

原来Java出了二十几年,.NET出了十五年了,到现在还有那么多人连如何高效地拼接字符串,什么时候该使用StringBuilder什么时候不该使用都不清楚。还有人说一万个字符串拼接用SB就快了等等,当然不是。SB的使用场景是目标字符串长度不确定的情况,和到底是十个还是十万个字符串,到底最终长度是一百还是一百万都没有关系。比如之前的场景,每个字符串都是确定的,最终长度自然也是确定的,这根本就不应该动用SB。假如你不知道应该怎么做,那只能让你去看下十五年前开始.NET就采取的做法。在目标字符串长度确定的情况下,出现目标字符串外任意一个额外的内存分配都是不及格。

另外有人说“一个”StringBuilder对象,好像没多少开销一样。但是SB不是一个对象,而是一串对象啊。你append过程中随时就会分配一个长度翻倍的新的char数组,然后还要复制一遍。多少人以为用SB就够了一样,都不知道需要指定一个capacity。
比如那段所谓被Java编译器“优化”的代码,capacity也没指定,翻倍和复制几乎肯定发生。当然假如他们知道指定capacity,也不会使用SB这么低效的做法了吧。

刚才忘记切换jdk了,java9 已经不是这样了,会使用makeConcatWithConstants 进行拼接

至于非final时用append来搞,从JDK 1.1时代就是这么玩了。当然,还是写javac的人偷懒,先确定长度再生成SB会好一些,但是碰到为null时怎么调用xxx.length(),搞到最后变成这样: 网页链接

正确的字符串拼接方式 网页链接 算出总长,分配目标字符串内存,把输入的字符串复制到正确的位置。出现任意额外的StringBuilder啊char数组分配什么的都是不及格。 ​​​​

我是真不知道直到 Java 9 才有 StringConcatFactory 这种东西的,所以才没能理解为什么 Java 程序员普遍觉得除了 .append 就只有 StringBuilder 一途

至于为什么 .NET 这边不鼓励滥用 StringBuilder,我觉得就算想一下为什么 Java 9 会有 StringConcatFactory 也该明白了。

多说一句,其实StringBuilder在拼接字符串时也不一定是最优的,因为它其实是把每次Append进去的东西复制展开,因此内存占用是和目标字符串长度相关的。有时候,你拿一个字符串数组/List保留输入字符串,最后用自己写的Concat(string[] input, beginIndex, length)拼起来,此时额外的内存占用就是和字符串数量相关,就远小于目标字符串长度了。而这个临时字符串数组甚至都可以复用,最终效果便又是零(额外)分配了。当然,这种方法并不是没有值得讨论的地方。一是实际开发时,复用一个巨型的StringBuilder,每线程一个(ThreadStatic),这可能也够了,开发起来也更方便(效果相对略差)。二是用StringBuilder时,每次Append的字符串可能可以被立即回收,而用上边描述的方法会导致字符串被长时间引用而被升代(假如它们本身就会被其他地方引用着那么自然就没这方面问题了)。总之内存优化时很多时候就是围绕字符串来的,毕竟字符串代表的是一大块连续内存,而且太容易生成新字符串(Split,ToUpper,Substring等等),操作起来要么浪费要么麻烦。更有甚者是,它很容易/已经被滥用了,例如目标函数就是要接受一个字符串时,你除了生成一个新的就没其他办法了。用Span<char>可以解决部分问题,但它一是太新用不上,二是也不能解决所有问题。收起全文

JEP280 为提高性能所做的努力 ​​​​

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容

  • Tip:笔者马上毕业了,准备开始 Java 的进阶学习计划。于是打算先从 String 类的源码分析入手,作为后面...
    石先阅读 12,010评论 16 58
  • 集合框架: 1)特点:存储对象;长度可变;存储对象的类型可不同2)Collection(1)List:有序的;元素...
    Demo_Yang阅读 1,258评论 0 4
  • 太乙翠溪苑,六个新同学。已是半夜深睡时,相约寻门去。一抹木碳火,两把签子肉,几瓶啤酒论相逢,归来亲如故。
    云水居士阅读 373评论 1 2
  • 从丛林到农田,我们用了几百万年,这不漫长,紧迫的战斗和进化从未停止。从农田到城市,我们只用了几十年,时间越来越不够...
    王子啊阅读 122评论 0 0
  • 猪爸坚强系列12: 一直以来XW联播中人民一片祥和、太平,生活在幸福中; 尤其有个大记者随便在街头逮个人问:你幸福...
    商业办公设计师木易阅读 202评论 0 0