朋的博客

          MySQL資料,Java技術(shù),管理思想,博弈論,Ajax,XP極限編程,H.264,HEVC,HDR
          隨筆 - 86, 文章 - 59, 評論 - 1069, 引用 - 0
          數(shù)據(jù)加載中……

          是String,StringBuffer還是StringBuilder?

          String StringBuffer 還是 StringBuilder

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

          ?????? 簡要的說, String 類型和 StringBuffer 類型的主要性能區(qū)別其實在于 String 是不可變的對象(為什么?問問 Java 的設(shè)計者吧,為什么 String 不是原生類型呢?)因此在每次對 String 類型進(jìn)行改變的時候其實都等同于生成了一個新的 String 對象,然后將指針指向新的 String 對象,所以經(jīng)常改變內(nèi)容的字符串最好不要用 String ,因為每次生成對象都會對系統(tǒng)性能產(chǎn)生影響,特別當(dāng)內(nèi)存中無引用對象多了以后, JVM GC 就會開始工作,那速度是一定會相當(dāng)慢的。這里嘗試舉個不是很恰當(dāng)?shù)睦樱?/span>

          ?????? String S1 = “abc”;

          ?????? For(int I = 0 ; I < 10000 ; I ++)? // For 模擬程序的多次調(diào)用

          ?????? {

          ????????????? S1 + = “def”;

          ????????????? S1 = “abc”;

          }

          如果是這樣的話,到這個 for 循環(huán)完畢后,如果內(nèi)存中的對象沒有被 GC 清理掉的話,內(nèi)存中一共有 萬個了,驚人的數(shù)目,而如果這是一個很多人使用的系統(tǒng),這樣的數(shù)目就不算很多了,所以大家使用的時候一定要小心。

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

          ?????? String S1 = “This is only a” + “ simple” + “ test”;

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

          ?????? 你會很驚訝的發(fā)現(xiàn),生成 String S1 對象的速度簡直太快了,而這個時候 StringBuffer 居然速度上根本一點都不占優(yōu)勢。其實這是 JVM 的一個把戲,在 JVM 眼里,這個

          ?????? String S1 = “This is only a” + “ simple” + “test”; 其實就是:

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

          String S2 = “This is only a”;

          String S3 = “ simple”;

          String S4 = “ test”;

          String S1 = S2 +S3 + S4;

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

          由此我們得到第一步結(jié)論:

          在大部分情況下 StringBuffer > String

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

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

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

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

          這樣說估計大家都能明白他們之間的區(qū)別了,那么下面我們再做一個一般性推導(dǎo):

          在大部分情況下 StringBuilder > StringBuffer

          因此,根據(jù)這個不等式的傳遞定理: 在大部分情況下

          StringBuilder > StringBuffer > String

          既然有這樣的推導(dǎo)結(jié)果了,我們做個測試驗證一下:

          測試代碼如下:

          public class testssb {

          ???

          ??? /** Creates a new instance of testssb */

          ??? final static int ttime = 10000;// 測試循環(huán)次數(shù)

          ??? public testssb() {

          ??? }

          ???

          ??? public void test(String s){

          ??????? long begin = System.currentTimeMillis();

          ??????? for(int i=0;i<ttime;i++){

          ??????????? s += "add";

          ??????? }

          ??????? long over = System.currentTimeMillis();

          ??????? System.out.println(" 操作 "+s.getClass().getName()+" 類型使用的時間為: "

          ??????????? + (over - begin) + " 毫秒 " );???????

          ??? }

          ??? public void test(StringBuffer s){

          ??????? long begin = System.currentTimeMillis();

          ??????? for(int i=0;i<ttime;i++){

          ??????????? s.append("add");

          ??????? }

          ??????? long over = System.currentTimeMillis();

          ??????? System.out.println(" 操作 "+s.getClass().getName()+" 類型使用的時間為: "

          ??????????? + (over - begin) + " 毫秒 " );???????

          ??? }

          ??? public void test(StringBuilder s){

          ??????? long begin = System.currentTimeMillis();

          ??????? for(int i=0;i<ttime;i++){

          ??????????? s.append("add");

          ??????? }

          ??????? long over = System.currentTimeMillis();

          ??????? System.out.println(" 操作 "+s.getClass().getName()+" 類型使用的時間為: "

          ??????????? + (over - begin) + " 毫秒 " );???????

          ??? }

          ??? // String 直接進(jìn)行字符串拼接的測試

          ??? public void test2(){

          ??????? String s2 = "abadf";

          ??????? long begin = System.currentTimeMillis();

          ??????? for(int i=0;i<ttime;i++){

          ??????????? String s = s2 + s2 + s2 ;

          ??????? }

          ??????? long over = System.currentTimeMillis();

          ??????? System.out.println(" 操作字符串對象引用相加類型使用的時間為: "

          ??????????? + (over - begin) + " 毫秒 " );???????

          ??? }

          ??? public void test3(){

          ??????? long begin = System.currentTimeMillis();

          ??????? for(int i=0;i<ttime;i++){

          ??????????? String s = "abadf" + "abadf" + "abadf" ;

          ??????? }

          ??????? long over = System.currentTimeMillis();

          ??????? System.out.println(" 操作字符串相加使用的時間為: "

          ??????????? + (over - begin) + " 毫秒 " );???????

          ?? ?}

          ???

          ??? public static void main(String[] args){

          ??? String s1 ="abc";

          ??? StringBuffer sb1 = new StringBuffer("abc");

          ??? StringBuilder sb2 = new StringBuilder("abc");

          ??? testssb t = new testssb();

          ??? t.test(s1);

          ??? t.test(sb1);

          ??? t.test(sb2);

          ?? ?t.test2();

          ??? t.test3();

          ??? }

          }

          以上代碼在 NetBeans 5.0 IDE/JDK1.6 上編譯通過

          循環(huán)次數(shù) ttime 10000 次的測試結(jié)果如下:

          操作 java.lang.String 類型使用的時間為: 4392 毫秒

          操作 java.lang.StringBuffer 類型使用的時間為: 0 毫秒

          操作 java.lang.StringBuilder 類型使用的時間為: 0 毫秒

          操作字符串對象引用相加類型使用的時間為: 15 毫秒

          操作字符串相加使用的時間為: 0 毫秒

          好像還看不出 StringBuffer StringBuilder 的區(qū)別,把 ttime 加到 30000 次看看:

          操作 java.lang.String 類型使用的時間為: 53444 毫秒

          操作 java.lang.StringBuffer 類型使用的時間為: 15 毫秒

          操作 java.lang.StringBuilder 類型使用的時間為: 15 毫秒

          操作字符串對象引用相加類型使用的時間為: 31 毫秒

          操作字符串相加使用的時間為: 0 毫秒

          StringBuffer StringBuilder 的性能上還是沒有太大的差異,再加大到 100000 看看,這里就不加入對 String 類型的測試了,因為對 String 類型這么大數(shù)據(jù)量的測試會很慢滴……

          操作 java.lang.StringBuffer 類型使用的時間為: 31 毫秒

          操作 java.lang.StringBuilder 類型使用的時間為: 16 毫秒

          能看出差別了,但其中有多次的測試結(jié)果居然是 StringBuffer StringBuilder 快,再加大一些到 1000000 看看(應(yīng)該不會當(dāng)機(jī)吧?):

          操作 java.lang.StringBuffer 類型使用的時間為: 265 毫秒

          操作 java.lang.StringBuilder 類型使用的時間為: 219 毫秒

          有些少區(qū)別了,而且結(jié)果很穩(wěn)定,再大點看看, ttime = 5000000

          ······ Exception in thread "main" java.lang.OutOfMemoryError: Java heap space ······

          呵呵,算了,不去測試了,基本來說都是在性能上都是 StringBuilder > StringBuffer > String 的了。

          其實我這里測試并不是很公平,因為都放在了一起以先后順序進(jìn)行,測試方法中間沒有考慮到JVM的GC收集前面產(chǎn)生的無引用對象垃圾而對執(zhí)行過程的中斷時間。如果大家有更好的想法或者思路歡迎跟我討論:chenpengyi#gmail.com。

          posted on 2006-05-04 01:15 benchensz 閱讀(25227) 評論(19)  編輯  收藏 所屬分類: 隨便寫寫-亂扯(基本不是有用的)

          評論

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          nice article, just one point to add:
          Instances of StringBuilder are not safe for use by multiple threads. If such synchronization is required then it is recommended that StringBuffer be used.
          2006-05-04 04:33 | Dedian

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          StringBuilder在多線程環(huán)境下是不安全的,需要同步的話就應(yīng)該采用StringBuffer,這點LZ已經(jīng)提到,受教,我對JDK5的了解和使用太少了。
          2006-05-04 11:31 | dennis

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          偶認(rèn)為這篇文章里至少有1個嚴(yán)重的錯誤
          String S1 = “abc”;

          For(int I = 0 ; I < 10000 ; I ++) // For 模擬程序的多次調(diào)用

          {

          S1 + = “def”;

          S1 = “abc”;

          }
          這段代碼運行后絕對不會是產(chǎn)生20000多個對象,因為S1 = “abc”; 在編譯的時候會產(chǎn)生一個靜態(tài)對象“abc”,當(dāng)循環(huán)執(zhí)行S1 = “abc”; 時不是產(chǎn)生一個String 對象,而是把那個靜態(tài)對象的句柄付給S1 。測試方法如下:
          S1 = “abc”;
          S2 = “abc”;
          if(S1==S2)說明S1和S2的句柄是一樣的,就是說指向同一個對象。大家可以試試!!!
          2006-05-11 22:35 | 土撥鼠

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          比較抱歉,確實是個很大的失誤。
          在寫這個的時候只是想證明對String對象的改變會導(dǎo)致生成很多新的對象,沒有考慮JVM對基本類型的處理方式。
          其實不僅僅是String,其他原生類型也一樣(char/int/double/float等)。如果在Stack中已經(jīng)存在了“abc”值,如果你需要再生成一個abc的對象,那么JVM只會生成一個指向原來在內(nèi)存中的“abc”值,而不是重新生成。
          確實是失誤,半夜的時候亂寫沒有考慮那么細(xì),已經(jīng)改了(2萬水分太大了,呵呵,不能虛報啊)。同時感謝小土同志的指出。
          2006-05-11 23:25 | 陳朋奕

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          上面的內(nèi)容讓我我很有收獲
          但我還有個疑問
          For(int I = 0 ; I < 10000 ; I ++) // For 模擬程序的多次調(diào)用
          {
          S1 + = “def”;
          S1 = “abc”;
          }

          S1+= "def"的時候會不會在堆中產(chǎn)生新的對象?
          我認(rèn)為這里因該會產(chǎn)生新對象,然后S1指向棧中常量池中的"abc",不知道我的理解是否正確?
          2006-05-12 21:53 | 劉竹軍

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          @陳朋奕
          其實這是 JVM 的一個把戲,在 JVM 眼里,這個

          String S1 = “This is only a” + “ simple” + “test”; 其實就是:

          String S1 = “This is only a simple test”;
          這里的表述不恰當(dāng)
          String S1 = “This is only a” + “ simple” + “test”; 其實就是:

          String S1 = “This is only a simple test”; 這么說是對的,但是如果是jvm的把戲,那么String S1 = “This is only a” + “ simple” + “test”; 是不是會和
          String S3=“This is only a”;
          S3+=“ simple”;
          S3+=“test”;
          System.out.println(S1==S3);的結(jié)果是什么的很顯然是false;

          其實這并不是JVM的把戲,而是java編譯器的把戲。
          給出如下源代碼
          package tt;
          public class TestString {

          /**
          * @param args
          */
          public static void main(String[] args) {
          // TODO Auto-generated method stub
          String s1 = "This is only a" + " simple " + "test";
          String s2 = "This is only a";
          s2+= " simple ";
          s2+= "test";
          System.out.println(s1==s2);
          }

          }
          那么它編譯(JDK5.0)后的class會是什么樣的呢,很遺憾我沒能用小穎反編譯,不過沒關(guān)系,我們用UltraEdit打開看看其二進(jìn)制代碼
          This is only a simple test
          This is only a
          simple
          test
          這么幾個字符串常量。
          也就是說是編譯器把"This is only a" + " simple " + "test";在編譯的時候變成了"This is only a simple test";
          但是,其中class中的另些細(xì)節(jié)讓我困惑:
          如下是從class中看到的
          前面的省略(里面的(.)是代表不能顯示的字節(jié))
          java/l
          ang/StringBuilde
          r......java/lang
          /String......val
          ueOf..&(Ljava/la
          ng/Object;)Ljava
          /lang/String;...
          ..........(Ljava
          /lang/String;)V.
          ............sim
          ple ......append
          后面的就不寫了。
          那是不是說在java編譯的時候,自動就將
          String s2 = "This is only a";
          s2+= " simple ";
          s2+= "test";
          用StringBulder來實現(xiàn)了呢,不解中。。。。。
          2006-05-13 14:26 | marmot

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          @劉竹軍兄弟
          會的,因為只要對象的值是stack中沒有的,那么就新生成。
          @marmot兄弟
          確實是Javac的問題,也是Javac優(yōu)化class的一個技巧。
          有新版小穎能反編譯JDK1.6的class文件嗎?
          2006-05-13 17:57 | 朋奕

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          我沒有找到!
          2006-05-14 20:17 | marmot

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          知道你在toronto大學(xué),留一下email地址,呵呵
          2006-05-14 20:49 | 朋奕

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          指定參數(shù)-Xms和-Xmx可以消除GC影響。但testsb類中寫法有問題,如果ttime超過一定數(shù)量級(譬如3000000)時指定命令:
          java -Xms50m -Xmx50m StringTest
          可以發(fā)現(xiàn)調(diào)用后面函數(shù)時會報錯OutOfMemoryError,解決方法是將StringBXX聲明到各個測試函數(shù)而不是main()中。
          2006-12-19 15:32 | fangread

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          請樓主試試這段代碼,好像和你說的相反。
          long c = System.currentTimeMillis();
          String test = null;
          for (int i = 0; i < 20000; i++) {
          test += String.valueOf(i);
          }
          long s = System.currentTimeMillis();
          System.out.println("String use time is:" + (s - c));
          System.out.println("StringBuilder start time is:" + s);
          StringBuilder sb = new StringBuilder();
          for (int i = 0; i < 20000; i++) {
          sb.append(String.valueOf(i));
          }
          long e = System.currentTimeMillis();
          System.out.println("StringBuilder end time is:" + e);
          System.out.println("StringBuilder use time is:" + (e - s));
          e = System.currentTimeMillis();
          System.out.println("StringBuffer start time is:" + e);
          StringBuffer buff = new StringBuffer();
          for (int i = 0; i < 20000; i++) {
          buff.append(String.valueOf(i));
          }
          long ea = System.currentTimeMillis();
          System.out.println("StringBuffer end time is:" + ea);
          System.out.println("StringBuffer use time is:" + (ea - e));

          我的運行結(jié)果是:

          String use time is:9515
          StringBuilder start time is:1184728951918
          StringBuilder end time is:1184728951934
          StringBuilder use time is:16
          StringBuffer start time is:1184728951950
          StringBuffer end time is:1184728951950
          StringBuffer use time is:0
          2007-07-18 11:26 | hbwjz_wolf

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          受教了~

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          wolf:
          不要混在一起,單獨運行每一個
          時間效果builder<Buffer<<String
          2007-11-29 14:57 | zly

          # re: 是String,StringBuffer還是StringBuilder?五   回復(fù)  更多評論   

          奇趣
          2008-09-12 00:32 | 冰涼

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          個人也認(rèn)同這個結(jié)果~ 畢竟非線程同步效率要高一點~
          2009-10-15 19:54 | xxwinnie

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          11212
          2010-08-03 17:23 | 1111

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          4545
          2010-08-03 17:30 | 1111

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          christian louboutin UKE33
          2010-12-31 14:15 | TT

          # re: 是String,StringBuffer還是StringBuilder?  回復(fù)  更多評論   

          寫的很好 受教
          2012-05-08 10:35 | 漸行漸遠(yuǎn)
          主站蜘蛛池模板: 渝中区| 厦门市| 陕西省| 左权县| 合阳县| 紫金县| 红河县| 响水县| 临沧市| 岱山县| 靖安县| 岗巴县| 花莲市| 平遥县| 肇州县| 灵川县| 云安县| 荔波县| 施甸县| 钦州市| 利津县| 平阳县| 永嘉县| 麦盖提县| 皮山县| 宜城市| 肇源县| 蓬溪县| 二连浩特市| 黑河市| 巴楚县| 涡阳县| 万山特区| 广饶县| 凤翔县| 晋州市| 建德市| 金溪县| 上饶县| 海南省| 岳池县|