程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA編程入門知識 >> java中String與StringBuilder的區別

java中String與StringBuilder的區別

編輯:JAVA編程入門知識

相信大家對 String 和 StringBuffer 的區別也已經很了解了,但是估計還是會有很多同志對這兩個類的工作原理有些不清楚的地方,今天我在這裡重新把這個概念給大家復習一下,順便牽出 J2SE 5.0 裡面帶來的一個新的字符操作的類—— StringBuilder (先別忙著扔我磚頭,我還算清醒,我這裡說的不是 C #, Java 也有 StringBuilder 類)。那麼這個 StringBuilder 和 StringBuffer 以及我們最早遇見的 String 類有那些區別呢?在不同的場合下我們應該用哪個呢?我講講自己對這幾個類的一點看法,也希望大家提出意見,每個人都有錯的地方,在錯了改的同時更是一個學習的好機會。

    簡要的說, String 類型和 StringBuffer 類型的主要性能區別其實在於 String 是不可變的對象(為什麼?問問 Java 的設計者吧,為什麼 String 不是原生類型呢?)因此在每次對 String 類型進行改變的時候其實都等同於生成了一個新的 String 對象,然後將指針指向新的 String 對象,所以經常改變內容的字符串最好不要用 String ,因為每次生成對象都會對系統性能產生影響,特別當內存中無引用對象多了以後, JVM 的 GC 就會開始工作,那速度是一定會相當慢的。這裡嘗試舉個不是很恰當的例子:

String S1 = “abc”;
For(int I = 0 ; I < 10000 ; I ++) // For 模擬程序的多次調用
{
S1 + = “def”;
S1 = “abc”;
}

    如果是這樣的話,到這個 for 循環完畢後,如果內存中的對象沒有被 GC 清理掉的話,內存中一共有 2 萬多個了,驚人的數目,而如果這是一個很多人使用的系統,這樣的數目就不算很多了,所以大家使用的時候一定要小心。

    而如果是使用 StringBuffer 類則結果就不一樣了,每次結果都會對 StringBuffer 對象本身進行操作,而不是生成新的對象,再改變對象引用。所以在一般情況下我們推薦使用 StringBuffer ,特別是字符串對象經常改變的情況下。而在某些特別情況下, String 對象的字符串拼接其實是被 JVM 解釋成了 StringBuffer 對象的拼接,所以這些時候 String 對象的速度並不會比 StringBuffer 對象慢,而特別是以下的字符串對象生成中, String 效率是遠要比 StringBuffer 快的:

String S1 = “This is only a” + “ simple” + “ test”;
StringBuffer Sb = new StringBuilder(“This is only a”).append(“ simple”).append(“ test”);

    你會很驚訝的發現,生成 String S1 對象的速度簡直太快了,而這個時候 StringBuffer 居然速度上根本一點都不占優勢。其實這是 JVM 的一個把戲,在 JVM 眼裡,這個

String S1 = “This is only a” + “ simple” + “test”; 其實就是: String S1 = “This is only a simple test”; 所以當然不需要太多的時間了。但大家這裡要注意的是,如果你的字符串是來自另外的 String 對象的話,速度就沒那麼快了,譬如:

String S2 = “This is only a”;
String S3 = “ simple”;
String S4 = “ test”;
String S1 = S2 +S3 + S4;

    這時候 JVM 會規規矩矩的按照原來的方式去做, S1 對象的生成速度就不像剛才那麼快了,一會兒我們可以來個測試作個驗證。

    由此我們得到第一步結論: 在大部分情況下 StringBuffer > String

    而 StringBuilder 跟他們比又怎麼樣呢?先簡單介紹一下, StringBuilder 是 JDK5.0 中新增加的一個類,它跟 StringBuffer 的區別看下面的介紹(來源 JavaWorld ):

    Java.lang.StringBuffer 線程安全的可變字符序列。類似於 String 的字符串緩沖區,但不能修改。可將字符串緩沖區安全地用於多個線程。可以在必要時對這些方法進行同步,因此任意特定實例上的所有操作就好像是以串行順序發生的,該順序與所涉及的每個線程進行的方法調用順序一致。

    每個字符串緩沖區都有一定的容量。只要字符串緩沖區所包含的字符序列的長度沒有超出此容量,就無需分配新的內部緩沖區數組。如果內部緩沖區溢出,則此容量自動增大。從 JDK 5.0 開始,為該類增添了一個單個線程使用的等價類,即 StringBuilder 。與該類相比,通常應該優先使用 StringBuilder 類,因為它支持所有相同的操作,但由於它不執行同步,所以速度更快。

    但是如果將 StringBuilder 的實例用於多個線程是不安全的。需要這樣的同步,則建議使用 StringBuffer 。

    這樣說估計大家都能明白他們之間的區別了,那麼下面我們再做一個一般性推導:

    在大部分情況下 StringBuilder > StringBuffer

    因此,根據這個不等式的傳遞定理: 在大部分情況下 StringBuilder > StringBuffer > String(操作次數越大,越穩定)。

獲取系統時間long start = System.currentTimeMillis(); long end = System.currentTimeMillis();便可知道運行的毫秒數值。
代碼如下:

import javax.swing.JOptionPane;
public class T1{
    public static void main(String args[]){
        String str;
        String str2;
        int i;
        StringBuffer sb=new StringBuffer();
        str=JOptionPane.showInputDialog(null,"輸入一個字符串");

        for(i=0;i<str.length()/2;i++)
            if(str.charAt(i)!=str.charAt(str.length()-i-1))
                break;

        if(i>=str.length()/2)
            JOptionPane.showMessageDialog(null,"是一個回文串");
        else
            JOptionPane.showMessageDialog(null,"不是一個回文串");

       

    }

}

忽略即非字母又非數字的字符,判斷回文串
代碼如下:

import javax.swing.JOptionPane;
 public class T2{
     public static void main(String args[]){
         String str;
         String str2;
         int i;
         StringBuffer sb=new StringBuffer();
         str=JOptionPane.showInputDialog(null,"輸入一個字符串");
         for(i=0;i<str.length();i++)
         {
             if(Character.isLetterOrDigit(str.charAt(i)))
                 sb.append(str.charAt(i));
         }
         str=sb.toString();
         str2=sb.reverse().toString();
         /*
         for(i=0;i<str.length()/2;i++)
             if(str.charAt(i)!=str.charAt(str.length()-i-1))
                 break;

         */
         if(str.equals(str2))
             JOptionPane.showMessageDialog(null,"是一個回文串");
         else
             JOptionPane.showMessageDialog(null,"不是一個回文串");

        

     }

 }

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved