国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Java 字符串拼接效率分析及最佳實踐

BakerJ / 2402人閱讀

摘要:兩個字符串拼接直接調(diào)用性能最好。關(guān)于的其他最佳實踐用時總是把能確定不為空的變量寫在左邊,如使用判斷空串,避免空指針異常。在需要把其他對象轉(zhuǎn)換為字符串對象時,使用而不是直接調(diào)用方法,因為前者已經(jīng)對空值進行檢測了,不會拋出空指針異常。

本文來源于問題 Java字符串連接最佳實踐?

java連接字符串有多種方式,比如+操作符,StringBuilder.append方法,這些方法各有什么優(yōu)劣(可以適當(dāng)說明各種方式的實現(xiàn)細節(jié))?

按照高效的原則,那么java中字符串連接的最佳實踐是什么?

有關(guān)字符串處理,都有哪些其他的最佳實踐?

廢話不多說,直接開始, 環(huán)境如下:

JDK版本: 1.8.0_65
CPU: i7 4790
內(nèi)存: 16G
直接使用+拼接

看下面的代碼:

    @Test
    public void test() {
        String str1 = "abc";
        String str2 = "def";
        logger.debug(str1 + str2);
    }

在上面的代碼中,我們使用加號來連接四個字符串,這種字符串拼接的方式優(yōu)點很明顯: 代碼簡單直觀,但是對比StringBuilderStringBuffer大部分情況下比后者都低,這里說是大部分情況下,我們用javap工具對上面代碼生成的字節(jié)碼進行反編譯看看在編譯器對這段代碼做了什么。

public void test();
    Code:
       0: ldc           #5                  // String abc
       2: astore_1
       3: ldc           #6                  // String def
       5: astore_2
       6: aload_0
       7: getfield      #4                  // Field logger:Lorg/slf4j/Logger;
      10: new           #7                  // class java/lang/StringBuilder
      13: dup
      14: invokespecial #8                  // Method java/lang/StringBuilder."":()V
      17: aload_1
      18: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      21: aload_2
      22: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      25: invokevirtual #10                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      28: invokeinterface #11,  2           // InterfaceMethod org/slf4j/Logger.debug:(Ljava/lang/String;)V
      33: return

從反編譯的結(jié)果來看,實際上對字符串使用+操作符進行拼接,編譯器會在編譯階段把代碼優(yōu)化成使用StringBuilder類,并調(diào)用append方法進行字符串拼接,最后調(diào)用toString方法,這樣看來是否可以認為在一般情況下其實直接使用+,反正編譯器也會幫我優(yōu)化為使用StringBuilder

StringBuilder源碼分析

答案自然是不可以的,原因就在于StringBuilder這個類它內(nèi)部做了些什么時。
我們看一看StringBuilder類的構(gòu)造器

    public StringBuilder() {
        super(16);
    }

    public StringBuilder(int capacity) {
        super(capacity);
    }

    public StringBuilder(String str) {
        super(str.length() + 16);
        append(str);
    }

    public StringBuilder(CharSequence seq) {
        this(seq.length() + 16);
        append(seq);
    }

StringBuilder提供了4個默認的構(gòu)造器, 除了無參構(gòu)造函數(shù)外,還提供了另外3個重載版本,而內(nèi)部都調(diào)用父類的super(int capacity)構(gòu)造方法,它的父類是AbstractStringBuilder,構(gòu)造方法如下:

    AbstractStringBuilder(int capacity) {
        value = new char[capacity];
    }

可以看到實際上StringBuilder內(nèi)部使用的是char數(shù)組來存儲數(shù)據(jù)(String、StringBuffer也是),這里capacity的值指定了數(shù)組的大小。結(jié)合StringBuilder的無參構(gòu)造函數(shù),可以知道默認的大小是16個字符。
也就是說如果待拼接的字符串總長度不小于16的字符的話,那么其實直接拼接和我們手動寫StringBuilder區(qū)別不大,但是我們自己構(gòu)造StringBuilder類可以指定數(shù)組的大小,避免分配過多的內(nèi)存。

現(xiàn)在我們再看看StringBuilder.append方法內(nèi)部做了什么事:

    @Override
    public StringBuilder append(String str) {
        super.append(str);
        return this;
    }

直接調(diào)用的父類的append方法

    public AbstractStringBuilder append(String str) {
        if (str == null)
            return appendNull();
        int len = str.length();
        ensureCapacityInternal(count + len);
        str.getChars(0, len, value, count);
        count += len;
        return this;
    }

在這個方法內(nèi)部調(diào)用了ensureCapacityInternal方法,當(dāng)拼接后的字符串總大小大于內(nèi)部數(shù)組value的大小時,就必須先擴容才能拼接,擴容的代碼如下:

    void expandCapacity(int minimumCapacity) {
        int newCapacity = value.length * 2 + 2;
        if (newCapacity - minimumCapacity < 0)
            newCapacity = minimumCapacity;
        if (newCapacity < 0) {
            if (minimumCapacity < 0) // overflow
                throw new OutOfMemoryError();
            newCapacity = Integer.MAX_VALUE;
        }
        value = Arrays.copyOf(value, newCapacity);
    }

StringBuilder在擴容時把容量增大到當(dāng)前容量的兩倍+2,這是很可怕的,如果在構(gòu)造的時候沒有指定容量,那么很有可能在擴容之后占用了浪費大量的內(nèi)存空間。其次擴容后還調(diào)用了Arrays.copyOf方法,這個方法把擴容前的數(shù)據(jù)復(fù)制到擴容后的空間內(nèi),這樣做的原因是:StringBuilder內(nèi)部使用char數(shù)組存放數(shù)據(jù),java的數(shù)組是不可擴容的,所以只能重新申請一片內(nèi)存空間,并把已有的數(shù)據(jù)復(fù)制到新的空間去,這里它最終調(diào)用了System.arraycopy方法來復(fù)制,這是一個native方法,底層直接操作內(nèi)存,所以比我們用循環(huán)來復(fù)制要塊的多,即便如此,大量申請內(nèi)存空間和復(fù)制數(shù)據(jù)帶來的影響也不可忽視。

使用+拼接和使用StringBuilder比較
    @Test
    public void test() {
        String str = "";
        for (int i = 0; i < 10000; i++) {
            str += "asjdkla";
        }
    }

上面這段代碼經(jīng)過優(yōu)化后相當(dāng)于:

    @Test
    public void test() {
        String str = null;
        for (int i = 0; i < 10000; i++) {
            str = new StringBuilder().append(str).append("asjdkla").toString();
        }
    }

一眼就能看出創(chuàng)建了太多的StringBuilder對象,而且在每次循環(huán)過后str越來越大,導(dǎo)致每次申請的內(nèi)存空間越來越大,并且當(dāng)str長度大于16時,每次都要擴容兩次!而實際上toString方法在創(chuàng)建String對象時,調(diào)用了Arrays.copyOfRange方法來復(fù)制數(shù)據(jù),此時相當(dāng)于每執(zhí)行一次,擴容了兩次,復(fù)制了3次數(shù)據(jù),這樣的代價是相當(dāng)高的。

    public void test() {
        StringBuilder sb = new StringBuilder("asjdkla".length() * 10000);
        for (int i = 0; i < 10000; i++) {
            sb.append("asjdkla");
        }
        String str = sb.toString();
    }

這段代碼的執(zhí)行時間在我的機器上都是0ms(小于1ms)和1ms,而上面那段代碼則大約在380ms!效率的差距相當(dāng)明顯。

同樣是上面的代碼,將循環(huán)次數(shù)調(diào)整為1000000時,在我的機器上,有指定capacity時耗時大約20ms,沒有指定capacity時耗時大約29ms,這個差距雖然和直接使用+操作符有了很大的提升(且循環(huán)次數(shù)增大了100倍),但是它依舊會觸發(fā)多次擴容和復(fù)制。

將上面的代碼改成使用StringBuffer,在我的機器上,耗時大約為33ms,這是因為StringBuffer在大部分方法上都加上了synchronized關(guān)鍵字來保證線程安全,執(zhí)行效率有一定程度上的降低。

使用String.concat拼接

現(xiàn)在再看這段代碼:

    @Test
    public void test() {
        String str = "";
        for (int i = 0; i < 10000; i++) {
            str.concat("asjdkla");
        }
    }

這段代碼使用了String.concat方法,在我的機器上,執(zhí)行時間大約為130ms,雖然直接相加要好的多,但是比起使用StringBuilder還要太多了,似乎沒什么用。其實并不是,在很多時候,我們只需要連接兩個字符串,而不是多個字符串的拼接,這個時候使用String.concat方法比StringBuilder要簡潔且效率要高。

    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);
    }

上面這段是String.concat的源碼,在這個方法中,調(diào)用了一次Arrays.copyOf,并且指定了len + otherLen,相當(dāng)于分配了一次內(nèi)存空間,并分別從str1和str2各復(fù)制一次數(shù)據(jù)。而如果使用StringBuilder并指定capacity,相當(dāng)于分配一次內(nèi)存空間,并分別從str1和str2各復(fù)制一次數(shù)據(jù),最后因為調(diào)用了toString方法,又復(fù)制了一次數(shù)據(jù)。

結(jié)論

現(xiàn)在根據(jù)上面的分析和測試可以知道:

Java中字符串拼接不要直接使用+拼接。

使用StringBuilder或者StringBuffer時,盡可能準(zhǔn)確地估算capacity,并在構(gòu)造時指定,避免內(nèi)存浪費和頻繁的擴容及復(fù)制。

在沒有線程安全問題時使用StringBuilder, 否則使用StringBuffer

兩個字符串拼接直接調(diào)用String.concat性能最好。

關(guān)于String的其他最佳實踐

equals時總是把能確定不為空的變量寫在左邊,如使用"".equals(str)判斷空串,避免空指針異常。

第二點是用來排擠第一點的.. 使用str != null && str.length() == 0來判斷空串,效率比第一點高。

在需要把其他對象轉(zhuǎn)換為字符串對象時,使用String.valueOf(obj)而不是直接調(diào)用obj.toString()方法,因為前者已經(jīng)對空值進行檢測了,不會拋出空指針異常。

使用String.format()方法對字符串進行格式化輸出。

在JDK 7及以上版本,可以在switch結(jié)構(gòu)中使用字符串了,所以對于較多的比較,使用switch代替if-else

我暫時想的起來的就這么幾個了.. 請大家?guī)兔ρa充補充...

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/66112.html

相關(guān)文章

  • Java日志正確使用姿勢

    摘要:但是往往越簡單的東西越容易讓我們忽視,從而導(dǎo)致一些不該有的發(fā)生,作為一名嚴(yán)謹(jǐn)?shù)某绦騿T,怎么能讓這種事情發(fā)生呢所以下面我們就來了解一下關(guān)于日志的那些正確使用姿勢。級別表示出現(xiàn)了嚴(yán)重錯誤,程序?qū)袛鄨?zhí)行。 前言 關(guān)于日志,在大家的印象中都是比較簡單的,只須引入了相關(guān)依賴包,剩下的事情就是在項目中盡情的打印我們需要的信息了。但是往往越簡單的東西越容易讓我們忽視,從而導(dǎo)致一些不該有的bug發(fā)...

    UCloud 評論0 收藏0
  • 再一次生產(chǎn) CPU 高負載排查實踐

    摘要:前言前幾日早上打開郵箱收到一封監(jiān)控報警郵件某某服務(wù)器負載較高,請研發(fā)盡快排查解決,發(fā)送時間正好是凌晨。其實早在去年我也處理過類似的問題,并記錄下來一次生產(chǎn)排查優(yōu)化實踐不過本次問題產(chǎn)生的原因卻和上次不太一樣,大家可以接著往下看。 showImg(https://segmentfault.com/img/remote/1460000019507452?w=1919&h=1080); 前言 ...

    kviccn 評論0 收藏0
  • 高性能Java代碼的最佳實踐

    摘要:高性能代碼的最佳實踐前言在這篇文章中,我們將討論幾個有助于提升應(yīng)用程序性能的方法。要獲得有關(guān)應(yīng)用程序需求的最好最可靠的方法是對應(yīng)用程序執(zhí)行實際的負載測試,并在運行時跟蹤性能指標(biāo)。 showImg(https://segmentfault.com/img/bVbtgk4?w=256&h=254); 高性能Java代碼的最佳實踐前言 在這篇文章中,我們將討論幾個有助于提升Java應(yīng)用程序性...

    stackfing 評論0 收藏0
  • GIAC 2017全球互聯(lián)網(wǎng)架構(gòu)大會最新日程

    摘要:月日至日,高可用架構(gòu)和聯(lián)合主辦的全球互聯(lián)網(wǎng)架構(gòu)大會將于上海光大會展中心舉行。全球互聯(lián)網(wǎng)架構(gòu)大會是高可用架構(gòu)技術(shù)社區(qū)推廣的面向架構(gòu)師技術(shù)負責(zé)人及高端技術(shù)從業(yè)人員的技術(shù)架構(gòu)大會。本次大會共有大板塊方向,場技術(shù)專題,個互聯(lián)網(wǎng)架構(gòu)案例。 showImg(https://segmentfault.com/img/bVZ3Vh?w=600&h=375);12月22日至23日,高可用架構(gòu)和msup聯(lián)...

    617035918 評論0 收藏0

發(fā)表評論

0條評論

BakerJ

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<