為什么 Java 8 中不需要 StringBuilder 拼接字符串
在Java開(kāi)發(fā)者中,字符串的拼接占用資源高往往是熱議的話(huà)題.
讓我們深入討論一下為什么會(huì)占用高資源。
在Java中,字符串對(duì)象是不可變的,意思是它一旦創(chuàng)建,你就無(wú)法再改變它。所以在我們拼接字符串的時(shí)候,創(chuàng)建了一個(gè)新的字符串,舊的被垃圾回收器所標(biāo)記。
如果我們處理上百萬(wàn)的字符串,然后,我們就會(huì)生成百萬(wàn)的額外字符串被垃圾回收器處理。
虛擬機(jī)底層在拼接字符串時(shí)執(zhí)行了眾多操作。拼接字符串最直接的點(diǎn)操作(dot operator)就是String#concat(String)操作。
public String concat(String str) { int otherLen = str.length(); if (otherLen == 0) { return this; } int len = value.length; char buf[] = Arrays.copyOf(value, len + otherLen); str.getChars(buf, len); return new String(buf, true); } public static char[] copyOf(char[] original, int newLength) { char[] copy = new char[newLength]; System.arraycopy(original, 0, copy, 0, Math.min(original.length, newLength)); return copy; } void getChars(char dst[], int dstBegin) { System.arraycopy(value, 0, dst, dstBegin, value.length); }
你可以看到一個(gè)字符數(shù)組被創(chuàng)建,長(zhǎng)度則是已有字符和拼接的字符長(zhǎng)度之和。然后,它們的值復(fù)制到新的字符數(shù)組中。最后,用這個(gè)字符數(shù)組創(chuàng)建一個(gè)String對(duì)象并返回。
所以這些操作繁多,如果你計(jì)算一下,會(huì)發(fā)現(xiàn)是O(n^2)的復(fù)雜度。
為了解決這個(gè)問(wèn)題,我們使用StringBuilder類(lèi)。它就像可變的String類(lèi)。拼接方法幫助我們避免不必要的復(fù)制。它擁有O(n)的復(fù)雜度,遠(yuǎn)遠(yuǎn)優(yōu)于O(n^2)。
然而Java 8默認(rèn)使用StringBuilder拼接字符串。
Java 8的文檔說(shuō)明:
為了提高字字符串拼接的性能,Java編譯器可以使用StringBuffer類(lèi)或類(lèi)似技術(shù),在使用求值表達(dá)式時(shí),減少中間String對(duì)象的創(chuàng)建。
Java編譯器處理這種情況:
public class StringConcatenateDemo { public static void main(String[] args) { String str = "Hello "; str += "world"; } }
上面的代碼會(huì)被編譯成如下字節(jié)碼:
public class StringConcatenateDemo { public StringConcatenateDemo(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object."<init>":()V 4: return public static void main(java.lang.String[]); Code: 0: ldc #2 // String Hello 2: astore_1 3: new #3 // class java/lang/StringBuilder 6: dup 7: invokespecial #4 // Method java/lang/StringBuilder."<init>":()V 10: aload_1 11: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 14: ldc #6 // String world 16: invokevirtual #5 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 19: invokevirtual #7 // Method java/lang/StringBuilder.toString:()Ljava/lang/String; 22: astore_1 23: return }
你可以在這些字節(jié)碼中看到,使用了StringBuilder。所以我們?cè)贘ava 8中不再需要使用StringBuilder類(lèi)。
英文原文:We Don't Need StringBuilder for Concatenation Anymore
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
SystemServer進(jìn)程啟動(dòng)過(guò)程解析
這篇文章主要為大家介紹了SystemServer進(jìn)程啟動(dòng)過(guò)程解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07Java ArrayList.add 的實(shí)現(xiàn)方法
這篇文章主要介紹了Java ArrayList.add 的實(shí)現(xiàn)方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-11-11SpringBoot配置log4j2的實(shí)現(xiàn)示例
SpringBoot中默認(rèn)使用Logback作為日志框架,本文主要介紹了SpringBoot配置log4j2的實(shí)現(xiàn)示例,具有一定的參考價(jià)值,感興趣的可以了解一下2023-12-12hashMap擴(kuò)容時(shí)應(yīng)該注意這些死循環(huán)問(wèn)題
今天給大家?guī)?lái)的是關(guān)于Java的相關(guān)知識(shí),文章圍繞著hashMap擴(kuò)容時(shí)的死循環(huán)問(wèn)題展開(kāi),文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06