Java字符串“+”你真的明白吗?

Java对字符串操作做了许多的优化,使用符号“+”来作为字符串拼接操作就是其中之一。

今天来抠一下这个东西的细节。

对于大部分Java开发来说,都知道Java会使用StringBuilder来优化字符串拼接操作。这种优化的一个极为重要的出发点就是,String在Java里面是一个不可变的对象,所谓的字符串拼接不过是用被拼接字符串的内容来创建一个新的字符串。

如果在Java没有优化的情况下,字符串拼接就变成不断创建新的String对象,每一个创建的对象都是一次拼接的结果。

StringBuilder可以减少这种开销。StringBuilder的原理非常接近ArrayList,即内部维持一个数组,在容量不足的情况下扩容。因此在使用StringBuilder的情况下,可以有效减少创建对象的次数。

那么问题来了:

  1. 是所有的字符串拼接都会被优化吗?
  2. 编译器做这种优化的时候,是如何确保线程安全的?

所有的字符串拼接都会被优化吗?

答案是YES,但是并不是所有的字符串拼接都会被同种方式——即使用StringBuilder——所优化。

这里存在一种更加强大的优化:编译器如果在编译期间就能确定字符串拼接的结果,那么编译器会将字符串拼接操作去掉,改为直接使用拼接后的结果——即编译器自身完成这个拼接操作。

实际上,这是编译器优化的一小部分工作。除了字符串拼接以外,还有数值计算,也会有类似的优化?;痪浠袄此担谙执嘁肫骼锩?,编译器会努力把计算提前做完——前提是它能够确切结算出来结果。与之类似的一个东西是Java的类加载过程会完成部分方法解析,即将方法调用指向真正的方法。这些体现的核心理念就是能在运行前完成的,就做完。

如:

字节码是:

也就是它实际上是直接使用hello world作为打印参数的值。

这里有意思的是,它在0,2,3,5的四条指令,实际上是可以忽略的。不过这并不是字符串拼接造成的,实际上是编译整体不够智能造成的。编译器其实在这个时候并没有断定后面除了用于字符串拼接以外,ab两个局部变量没有再使用过。所以编译器只能非常保守的继续保留着四条指令。

这四条指令在JIT阶段有极大的可能会被优化掉。不过那都是在运行期的时候了。

另外,是否注意到图中我将两个局部变量都声明成了final了。那是因为,只有声明成final,编译器才能确定该变量的值,并且可以肯定这个变量的值在拼接操作并未被修改过。

如果没有final关键字,那么会变成:

也就是使用StringBuiler

这里我要额外讨论一个所谓的事实final变量。在Java里面,最开始使用内部匿名类的时候,内部匿名类要使用外部变量,那么只能将该外部变量声明成final。

否则编译器会报错。

直到后来(忘了是哪个版本,好像是Java8引入lambda表达式的时候),如果编译器确定你这个变量中途并未被修改过,那么即便不声明成final都可以在内部匿名类使用。

这就是所谓的事实final变量。这个名词是我杜撰的,专业的说法不知道叫什么。

所以理论上,编译器是完全可以断定在这个过程中字符串变量有没有被修改过,而后执行这种优化的。很可惜,编译器并没有利用这个信息。这是我一直觉得稍微有点遗憾的地方。

StringBuilder的优化是如何保证线程安全的?

这是一个看起来没什么营养的问题,一思考又觉得很有营养的问题,考虑清楚之后终于确定的确没什么营养的问题。

答案是,其实它不保证线程安全,它只是保证和不用StringBuilder优化时候的语义一致。这就是指,如果不是用StringBuiler的地方线程不安全,那么使用StringBuilder优化也不安全。

首先,大部分优化是安全的,这种线程安全的第一条保证是:String是不可变类型。这个无需多说,稍微思考一下就知道的。

再深入一点的话,如前面的例子,因为字符串只出现在方法里面,是作为方法的局部变量出现的,所以天然是线程安全的。

那么,如果我的代码是这样的呢?

这是一个初看起来会线程安全的代码,实际上却并没有的代码。

先来分析staticC。staticC是在类初始化的时候完成计算的。JVM的类加载机制可以确保,对于一个类加载器来说,staticC那句代码,只会被一个线程执行。在完成类初始化完成之前,staticAstaticB是无法被修改的。所以这个可以保证是线程安全的。

而变量c就要复杂一点了,理论上,c会在调用构造初始化方法之前完成初始化。如果从字节码的角度来解释,就是c会在构造方法里面的任何代码执行之前完成。

然而创建对象,在JVM层面上并不是一个原子步骤,它大概是有两步:

  1. new指令执行,大体上可以理解为分配内存;
  2. 调用初始化方法<init>;

所以在JIT的情况下,可能第一步执行完之后,引用就被外部获取了,这个时候他们就可能并发修改变量a或者b的值:

  1. new指令执行,大体上可以理解为分配内存;
  2. a或者b的值被修改---JIT情况下;
  3. 调用初始化方法<init>;

所以我才说,这种优化只保证和没有优化的语义一致。和线程安全没什么关系。

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

推荐阅读更多精彩内容

  • 《Java从小白到大?!分街拾嬉丫霞芰耍。?! 由字符组成的一串字符序列,称为“字符串”,在前面的章节中也多次用到...
    tony关东升阅读 808评论 0 2
  • 九种基本数据类型的大小,以及他们的封装类。(1)九种基本数据类型和封装类 (2)自动装箱和自动拆箱 什么是自动装箱...
    关玮琳linSir阅读 1,882评论 0 47
  • 相关概念 面向对象的三个特征 封装,继承,多态.这个应该是人人皆知.有时候也会加上抽象. 多态的好处 允许不同类对...
    东经315度阅读 1,936评论 0 8
  • 由于时间仓促,有些地方未写完,后面会继续补充.如有不妥之处,欢迎及时与我沟通. 如果你也是在学习java,给你们推...
    分不清java阅读 2,829评论 0 15
  • 山林中弯弯曲曲的道路,绕绕弯弯,如果用相机拍摄下来,也许是一幅美丽的画卷;蓝天白云下,青山矗立,逶迤的小道再加几只...
    柒而漆阅读 138评论 0 0