當前位置 : IT培訓網 > Java開發 > Java教程 > Java語言StringBuffer與StringBuider有啥區別

Java語言StringBuffer與StringBuider有啥區別

時間:2016-08-03 16:04:25??來源:Java培訓網??作者:IT培訓網??已有:名學員訪問該課程
在Java學習過程中,我們常常遇到String、StringBuffer、StringBuilder這些類,其實它們都是java編程中經常使用的字符串類,他們之間的區別也是經常在面試中會問到的問題。現在總結一下,看看他們的不同與相同。

在Java學習過程中,我們常常遇到String、StringBuffer、StringBuilder這些類,其實它們都是java編程中經常使用的字符串類,他們之間的區別也是經常在面試中會問到的問題。現在總結一下,看看他們的不同與相同。

String 的值是不可變的,每次對String的操作都會生成新的String對象,不僅效率低,而且耗費大量內存空間。

StringBuffer類和String類一樣,也用來表示字符串,但是StringBuffer的內部實現方式和String不同,在進行字符串處理時,不生成新的對象,在內存使用上要優于String。

StringBuffer 默認分配16字節長度的緩沖區,當字符串超過該大小時,會自動增加緩沖區長度,而不是生成新的對象。

StringBuffer不像String,只能通過 new 來創建對象,不支持簡寫方式,例如:

StringBuffer str1 = new StringBuffer();  // 分配16個字節長度的緩沖區

StringBuffer str2 = =new StringBuffer(512);  // 分配512個字節長度的緩沖區

// 在緩沖區中存放了字符串,并在后面預留了16個字節長度的空緩沖區

StringBuffer str3 = new StringBuffer("www.straightshotcamera.com");

StringBuffer類的主要方法

StringBuffer類中的方法主要偏重于對于字符串的操作,例如追加、插入和刪除等,這個也是StringBuffer類和String類的主要區別。實際開發中,如果需要對一個字符串進行頻繁的修改,建議使用 StringBuffer。

1) append() 方法

append() 方法用于向當前字符串的末尾追加內容,類似于字符串的連接。調用該方法以后,StringBuffer對象的內容也發生改變,例如:

StringBuffer str = new StringBuffer(“biancheng100”);

str.append(true);

則對象str的值將變成”biancheng100true”。注意是str指向的內容變了,不是str的指向變了。

字符串的”+“操作實際上也是先創建一個StringBuffer對象,然后調用append()方法將字符串片段拼接起來,最后調用toString()方法轉換為字符串。

 

這樣看來,String的連接操作就比StringBuffer多出了一些附加操作,效率上必然會打折扣。

但是,對于長度較小的字符串,”+“操作更加直觀,更具可讀性,有些時候可以稍微犧牲一下效率。

2)  deleteCharAt()

deleteCharAt() 方法用來刪除指定位置的字符,并將剩余的字符形成新的字符串。例如:

StringBuffer str = new StringBuffer("abcdef");

str. deleteCharAt(3);

該代碼將會刪除索引值為3的字符,即”d“字符。

 

你也可以通過delete()方法一次性刪除多個字符,例如:

StringBuffer str = new StringBuffer("abcdef");

str.delete(1, 4);

該代碼會刪除索引值為1~4之間的字符,包括索引值1,但不包括4。

3) insert() 方法

insert() 用來在指定位置插入字符串,可以認為是append()的升級版。例如:

StringBuffer str = new StringBuffer("abcdef");

str.insert(3, "xyz");

最后str所指向的字符串為 abcdxyzef。

4) setCharAt() 方法

setCharAt() 方法用來修改指定位置的字符。例如:

StringBuffer str = new StringBuffer("abcdef");

str.setCharAt(3, 'z');

該代碼將把索引值為3的字符修改為 z,最后str所指向的字符串為 abczef。

以上僅僅是部分常用方法的簡單說明,更多方法和解釋請查閱API文檔。

String和StringBuffer的效率對比

為了更加明顯地看出它們的執行效率,下面的代碼,將26個英文字母加了10000次。

public class Demo {

    public static void main(String[] args){

        String fragment = "abcdefghijklmnopqrstuvwxyz";

        int times = 10000;

      

        // 通過String對象

        long timeStart1 = System.currentTimeMillis();

        String str1 = "";

        for (int i=0; i<times; i++) {

            str1 += fragment;

        }

        long timeEnd1 = System.currentTimeMillis();

        System.out.println("String: " + (timeEnd1 - timeStart1) + "ms");

      

        // 通過StringBuffer

        long timeStart2 = System.currentTimeMillis();

        StringBuffer str2 = new StringBuffer();

        for (int i=0; i<times; i++) {

            str2.append(fragment);

        }

        long timeEnd2 = System.currentTimeMillis();

        System.out.println("StringBuffer: " + (timeEnd2 - timeStart2) + "ms");

    }

}

運行結果:

String: 5287ms

StringBuffer: 3ms

結論很明顯,StringBuffer的執行效率比String快上千倍,這個差異隨著疊加次數的增加越來越明顯,當疊加次數達到30000次的時候,運行結果為:

String: 35923ms

StringBuffer: 8ms

所以,強烈建議在涉及大量字符串操作時使用StringBuffer。

StringBuilder

StringBuilder類和StringBuffer類功能基本相似,方法也差不多,主要區別在于StringBuffer類的方法是多線程安全的,而StringBuilder不是線程安全的,相比而言,StringBuilder類會略微快一點。

StringBuffer、StringBuilder、String中都實現了CharSequence接口。

CharSequence是一個定義字符串操作的接口,它只包括length()、charAt(int index)、subSequence(int start, int end) 這幾個API。

 

StringBuffer、StringBuilder、String對CharSequence接口的實現過程不一樣,如下圖所示:

Java語言StringBuffer與StringBuider有啥區別_www.straightshotcamera.com

圖1  對CharSequence接口的實現

可見,String直接實現了CharSequence接口;StringBuilder 和 StringBuffer都是可變的字符序列,它們都繼承于AbstractStringBuilder,實現了CharSequence接口。

總結

線程安全:

StringBuffer:線程安全

StringBuilder:線程不安全

速度:

一般情況下,速度從快到慢為 StringBuilder > StringBuffer > String,當然這是相對的,不是絕對的。

使用環境:

操作少量的數據使用 String;

單線程操作大量數據使用 StringBuilder;

多線程操作大量數據使用 StringBuffer。

頂一下
(0)
0%
踩一下
(0)
0%
------分隔線----------------------------
------分隔線----------------------------
Java教程
1、Java 概述
1.1 Java語言概述
1.2 Java虛擬機以及跨平臺原理
1.3 Java的主要就業方向
1.4 Java的不同版本
1.5 Java開發環境搭建
1.6 第一個Java程序示例
1.7 Java類和對象的概念
1.8 Java類庫及其組織結構
1.9 Java import
2、Java 語法基礎
2.1 Java數據類型以及變量的定義
2.2 Java數據類型轉換
2.3 Java運算符
2.4 Java流程控制
2.5 Java數組的定義和使用
2.6 Java字符串(String)
2.7 Java StringBuffer與StringBuider
2.8 強調一下編程風格
3、Java 類與對象
3.1 Java類的定義及其實例化
3.2 Java訪問修飾符
3.3 Java變量的作用域
3.4 Java this關鍵字詳解
3.5 Java方法重載
3.6 Java類的基本運行順序
3.7 Java包裝類、拆箱和裝箱詳解
3.8 再談Java包
3.9 源文件的聲明規則
4、Java 繼承和多態
4.1 繼承的概念與實現
4.2 Java super關鍵字
4.3 繼承中的方法的覆蓋和重載
4.4 多態和動態綁定
4.5 instanceof 運算符
4.6 多態對象的類型轉換
4.7 Java static關鍵字
4.8 Java final關鍵字
4.9 Java Object類
5、面向對象高級特性
5.1 Java內部類及其實例化
5.2 內部類的分類
5.3 抽象類的概念和使用
5.4 接口的概念和使用
5.5 接口和抽象類的區別
5.6 Java 泛型
5.7 泛型通配符和類型參數的范圍
6、異常處理
6.1 異常處理基礎
6.2 異常類型Java語言中常見的異常類型有哪些
6.3 未被捕獲的異常
6.4 try和catch的使用
6.5 多重catch語句的使用
6.6 try語句的嵌套
6.7 throw:異常的拋出
6.8 throws子句
6.9 finally塊
6.10 Java的內置異常
6.11 創建自己的異常子類
6.12 斷言
7、線程編程
7.1 線程的概念
7.2 Java線程模型
7.3 主線程
7.4 創建線程
7.5 創建多線程
7.6 isAlive()和join()的使用
7.7 線程優先級
7.8 線程同步
7.9 線程間通信
7.10 線程死鎖
7.11 線程的掛起、恢復和終止
8、輸入輸出(IO)操作
8.1 輸入輸出基本概念
8.2 面向字符的輸入流
8.3 面向字符的輸出流
8.4 面向字節的輸入輸出流
8.5 面向字節流的應用
8.6 文件與目錄管理
8.7 文件的隨機讀寫
8.8 文件的壓縮處理
9、常用類庫、向量與哈希
9.1 Java基礎類庫
9.2 Object類
9.3 Java語言包(java.lang)簡介
9.4 日期和時間類
9.5 向量及其應用
9.6 哈希表及其應用
10、圖形界面(GUI)設計
10.1 圖形界面設計基礎
10.2 框架窗口
10.3 標簽、按鈕和按鈕事件
10.4 面板
10.5 布局設計
10.6 文本框和文本區
10.7 文本框和文本區的輸入輸出
10.8 選擇框和單選按鈕
10.9 列表和組合框
10.10 菜單
10.11 對話框
10.12 滾動條
10.13 鼠標事件
10.14 鍵盤事件
11、圖形、圖像與多媒體
11.1 繪圖基礎
11.2 設置字型和顏色
11.3 繪圖模式
11.4 Graphics類的繪圖方法
11.5 Graphics2D類的繪圖方法
11.6 圖像處理基礎
11.7 圖像緩沖技術
11.8 多媒體基礎
12、網絡與數據庫編程
12.1 IP地址和InetAddress類
12.2 統一資源定位符
12.3 套接字(Socket)
12.4 數據庫連接
12.5 幾個重要的類和接口
12.6 數據庫查詢
12.7 數據庫更新
12.8 插入記錄
12.9 修改記錄
12.10 刪除記錄
激情色播