自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

深入了解Java中的StringBuilder與StringBuffer

開發(fā)
昨天在講解深入解析 String.intern() 提到了StringBuilder,那么今天就來(lái)講解一下。

1. StringBuffer和StringBuilder的區(qū)別

因?yàn)樽址豢勺?,?dāng)字符串拼接(尤其是使用+號(hào)操作符)時(shí),需要考量性能的問(wèn)題,不多毫無(wú)顧忌的創(chuàng)建太多String對(duì)象,從而對(duì)內(nèi)存造成不必要壓力。

因此Java專門設(shè)計(jì)StringBuilder類來(lái)解決該問(wèn)題

public final class StringBuffer extends AbstractStringBuilder implements Serializable, CharSequence {

    public StringBuffer() {
        super(16);
    }
    
    public synchronized StringBuffer append(String str) {
        super.append(str);
        return this;
    }

    public synchronized String toString() {
        return new String(value, 0, count);
    }

    //...方法
}

從上面代碼我們可以發(fā)現(xiàn)StringBuffer在進(jìn)行字符串操作時(shí),方法都添加上synchronized關(guān)鍵字進(jìn)行同步,這主要是考慮到多線程環(huán)境下安全問(wèn)題。因?yàn)榧恿藄ynchronized,所以在非多線程下,執(zhí)行效率就會(huì)比較低,這是添加了沒(méi)必要的鎖。

考慮到性能問(wèn)題,Java又給StringBuffer添加了一個(gè)孿生兄弟StringBuilder在方法上沒(méi)有添加synchronized關(guān)鍵字,因此無(wú)論單線程還是多線程效率都會(huì)高。

public final class StringBuilder extends AbstractStringBuilder
    implements java.io.Serializable, CharSequence
{
    // ...

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

    public String toString() {
        // Create a copy, don't share the array
        return new String(value, 0, count);
    }

    // ...其他方法
}

是因?yàn)榉椒ㄉ蠜](méi)有synchronized關(guān)鍵字,所以StringBuilder多線程情況不安全 ,如果要在多線程環(huán)境下修改字符串,你到時(shí)候可以使用 ThreadLocal 來(lái)避免多線程沖突。

public class ThreadSafeStringBuilder {
    // 使用ThreadLocal為每個(gè)線程提供獨(dú)立的StringBuilder對(duì)象
    private static final ThreadLocal<StringBuilder> threadLocalStringBuilder = ThreadLocal.withInitial(StringBuilder::new);

    public static void appendString(String str) {
        // 獲取當(dāng)前線程的StringBuilder對(duì)象
        StringBuilder stringBuilder = threadLocalStringBuilder.get();
        // 在StringBuilder對(duì)象上執(zhí)行字符串拼接操作
        stringBuilder.append(str);
    }

    public static String getString() {
        // 獲取當(dāng)前線程的StringBuilder對(duì)象
        StringBuilder stringBuilder = threadLocalStringBuilder.get();
        // 返回StringBuilder對(duì)象的字符串表示
        return stringBuilder.toString();
    }

    public static void main(String[] args) {
        // 創(chuàng)建多個(gè)線程并發(fā)執(zhí)行字符串拼接操作
        Runnable task = () -> {
            for (int i = 0; i < 10; i++) {
                appendString(Thread.currentThread().getName() + "-" + i + " ");
            }
            // 輸出當(dāng)前線程的字符串結(jié)果
            System.out.println(Thread.currentThread().getName() + ": " + getString());
            // 清空當(dāng)前線程的StringBuilder對(duì)象,以便下次使用
            threadLocalStringBuilder.get().setLength(0);
        };

        // 啟動(dòng)多個(gè)線程
        Thread[] threads = new Thread[5];
        for (int i = 0; i < threads.length; i++) {
            threads[i] = new Thread(task);
            threads[i].start();
        }

        // 等待所有線程執(zhí)行完成
        for (Thread thread : threads) {
            try {
                thread.join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}

注意:實(shí)際開發(fā)中,StringBuilder 的使用頻率也是遠(yuǎn)高于 StringBuffer,甚至可以說(shuō),StringBuilder 完全取代了 StringBuffer。

2. StringBuilder使用

在深入解析 String.intern() 說(shuō)過(guò)編譯器遇到 + 號(hào)這個(gè)操作符的時(shí)候,會(huì)將 new String("spring") + new String("葵花寶典") 編譯代碼如下:

new StringBuilder().append("spring").append("葵花寶典").toString();

雖然過(guò)程我們看不見(jiàn),這正是 Java 的只能之處,Java可以在編譯的時(shí)幫我們做很多優(yōu)化,這樣既可以提高我們的開發(fā)效率(+ 號(hào)寫起來(lái)比創(chuàng)建 StringBuilder 對(duì)象便捷得多),也不會(huì)影響 JVM 的執(zhí)行效率。

如果我們使用 javap 反編譯 new String("spring") + new String("葵花寶典") 的字節(jié)碼的時(shí)候,也是能看出 StringBuilder 的影子。

0: new           #2                  // class java/lang/StringBuilder
       3: dup
       4: invokespecial #3                  // Method java/lang/StringBuilder."<init>":()V
       7: new           #4                  // class java/lang/String
      10: dup
      11: ldc           #5                  // String spring
      13: invokespecial #6                  // Method java/lang/String."<init>":(Ljava/lang/String;)V
      16: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      19: new           #4                  // class java/lang/String
      22: dup
      23: ldc           #8                  // String 葵花寶典
      25: invokespecial #6                  // Method java/lang/String."<init>":(Ljava/lang/String;)V
      28: invokevirtual #7                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
      31: invokevirtual #9                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
      34: astore_1
      35: return

可以發(fā)現(xiàn)Java 編譯器將字符串拼接操作(+)轉(zhuǎn)換為了 StringBuilder 對(duì)象的 append 方法,然后再調(diào)用 StringBuilder 對(duì)象的 toString 方法返回拼接后的字符串。

3. StringBuilder的內(nèi)部實(shí)現(xiàn)

3.1.  StringBuilder的toString()方法

public String toString() {
    return new String(value, 0, count);
}

value 是一個(gè) char 類型的數(shù)組

/**
 * The value is used for character storage.
 */
char[] value;

StringBuilder創(chuàng)建對(duì)象是,會(huì)給value分配內(nèi)存空間(初始容量16),來(lái)存儲(chǔ)字符串。

public StringBuilder() {
    super(16);
}

隨著字符串不斷拼接,value數(shù)組長(zhǎng)度會(huì)自動(dòng)進(jìn)行擴(kuò)容操作,將字符數(shù)組長(zhǎng)度增加到足夠容納新字符串的大小。value動(dòng)態(tài)擴(kuò)容的過(guò)程類似于ArrayList中的擴(kuò)容機(jī)制,確保了在拼接大量字符串時(shí)的高效性

3.2.  StringBuilder的append(String str) 方法

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

StringBuilder類的append(String str) 方法實(shí)際調(diào)用AbstractStringBuilder類中append方法。該方法會(huì)檢查當(dāng)前字符序列中的字符是否夠用,如果不夠用則會(huì)進(jìn)行擴(kuò)容,并將指定字符串追加到字符序列的末尾。

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

AbstractStringBuilder類的append(String str) 方法將指定的字符串追加到當(dāng)前字符序列中。如果指定字符串為 null,則追加字符串 "null";否則,該方法會(huì)檢查指定字符串的長(zhǎng)度,根據(jù)當(dāng)前字符序列中已有字符的數(shù)量以及指定字符串的長(zhǎng)度來(lái)判斷是否需要擴(kuò)容。如果需要擴(kuò)容,則會(huì)分配一個(gè)新的字符數(shù)組,將原有字符序列的內(nèi)容復(fù)制到新的字符數(shù)組中,并將指定字符串的內(nèi)容追加到新字符數(shù)組的末尾。這樣就確保了在追加字符串時(shí),字符序列的容量始終能夠滿足當(dāng)前字符數(shù)量的需求,避免了不必要的內(nèi)存浪費(fèi)說(shuō)明:擴(kuò)容調(diào)用方法ensureCapacityInternal(int minimumCapacity)方法,擴(kuò)容之后,將指定字符串的字符拷貝到字符序列中。

3.3. AbstractStringBuilder的ensureCapacityInternal(int minimumCapacity)方法

private void ensureCapacityInternal(int minimumCapacity) {
    // overflow-conscious code
    int oldCapacity = value.length >> coder;
    if (minimumCapacity - oldCapacity > 0) {
        value = Arrays.copyOf(value,
                newCapacity(minimumCapacity) << coder);
    }
}

private int newCapacity(int minCapacity) {
    // overflow-conscious code
    int oldCapacity = value.length >> coder;
    int newCapacity = (oldCapacity << 1) + 2;
    if (newCapacity - minCapacity < 0) {
        newCapacity = minCapacity;
    }
    int SAFE_BOUND = MAX_ARRAY_SIZE >> coder;
    return (newCapacity <= 0 || SAFE_BOUND - newCapacity < 0)
        ? hugeCapacity(minCapacity)
        : newCapacity;
}

ensureCapacityInternal(int minimumCapacity) 方法用于確保當(dāng)前字符序列的容量至少等于指定的最小容量 minimumCapacity。如果當(dāng)前容量小于指定的容量,就會(huì)為字符序列分配一個(gè)新的內(nèi)部數(shù)組。新容量的計(jì)算方式如下:

  • 如果指定的最小容量大于當(dāng)前容量,則新容量為兩倍的舊容量加上 2。為什么要加 2 呢?這是因?yàn)樵谀承┣闆r下,僅僅將容量加倍可能仍然不足以容納更多的字符。例如,對(duì)于非常小的字符串(比如空的或只有一個(gè)字符的 StringBuilder),僅僅將容量加倍可能仍然不足以容納更多的字符。因此,加上 2 提供了一個(gè)最小的增長(zhǎng)量,確保即使對(duì)于很小的初始容量,擴(kuò)容后也能至少添加一些字符而不需要立即再次擴(kuò)容。
  • 如果指定的最小容量小于等于當(dāng)前容量,則不會(huì)進(jìn)行擴(kuò)容,直接返回當(dāng)前對(duì)象。這樣做是為了避免不必要的內(nèi)存浪費(fèi)和性能開銷。

3.4 StringBuilder的 reverse 方法

public StringBuilder reverse() {
    super.reverse();
    return this;
}

StringBuilder類的reverse() 方法實(shí)際調(diào)用AbstractStringBuilder類中reverse()方法。該方法會(huì)檢查當(dāng)前字符序列中的字符是否夠用,如果不夠用則會(huì)進(jìn)行擴(kuò)容,并將指定字符串追加到字符序列的末尾。

public AbstractStringBuilder reverse() {
        byte[] val = this.value;
        int count = this.count;
        int coder = this.coder;
        int n = count - 1; // 字符序列的最后一個(gè)字符的索引
        if (COMPACT_STRINGS && coder == LATIN1) {
            for (int j = (n-1) >> 1; j >= 0; j--) {
                int k = n - j; // 計(jì)算相對(duì)于 j 對(duì)稱的字符的索引
                byte cj = val[j];  // 獲取當(dāng)前位置的字符
                val[j] = val[k]; // 交換字符
                val[k] = cj; // 交換字符
            }
        } else {
            StringUTF16.reverse(val, count);
        }
        return this;  // 返回反轉(zhuǎn)后的字符串構(gòu)建器對(duì)象
    }

1.初始化:

n表示字符串中最后一個(gè)字符索引

2.字符串反轉(zhuǎn):

  • 方法通過(guò)一個(gè) for 循環(huán)遍歷字符串的前半部分和后半部分,這是一個(gè)非常巧妙的點(diǎn),比從頭到尾遍歷省了一半的時(shí)間。(n-1) >> 1 是 (n-1) / 2 的位運(yùn)算表示,也就是字符串的前半部分的最后一個(gè)字符的索引。
  • 在每次迭代中,計(jì)算出與當(dāng)前索引 j 對(duì)稱的索引 k,并交換這兩個(gè)索引位置的字符。
責(zé)任編輯:華軒 來(lái)源: springboot葵花寶典
相關(guān)推薦

2019-11-29 16:21:22

Spring框架集成

2017-01-20 08:30:19

JavaScriptfor循環(huán)

2020-07-20 06:35:55

BashLinux

2010-11-19 16:22:14

Oracle事務(wù)

2010-07-13 09:36:25

2010-06-23 20:31:54

2009-08-25 16:27:10

Mscomm控件

2022-08-26 13:48:40

EPUBLinux

2020-09-21 09:53:04

FlexCSS開發(fā)

2023-10-13 00:09:20

桶排序排序算法

2018-02-24 13:21:02

2013-04-10 11:16:19

iPad的MouseE

2016-10-20 08:46:17

2018-09-04 16:20:46

MySQ索引數(shù)據(jù)結(jié)構(gòu)

2021-09-03 08:27:47

FortinetSASE平臺(tái)安全

2023-11-02 07:55:31

Python對(duì)象編程

2023-12-01 09:14:58

ReactFiber

2019-08-02 08:59:21

Token認(rèn)證服務(wù)器

2023-10-08 00:02:07

Java排序算法

2024-04-17 09:01:08

Python深拷貝淺拷貝
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)