<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    love fish大鵬一曰同風起,扶搖直上九萬里

    常用鏈接

    統(tǒng)計

    積分與排名

    friends

    link

    最新評論

    String,StringBuffer,StringBuilder(轉(zhuǎn)載)

           簡要的說, String 類型和 StringBuffer 類型的主要性能區(qū)別其實在于 String 是不可變的對象(為什么?問問 Java 的設(shè)計者吧,為什么 String 不是原生類型呢?)因此在每次對 String 類型進行改變的時候其實都等同于生成了一個新的 String 對象,然后將指針指向新的 String 對象,所以經(jīng)常改變內(nèi)容的字符串最好不要用 String ,因為每次生成對象都會對系統(tǒng)性能產(chǎn)生影響,特別當內(nèi)存中無引用對象多了以后, JVM GC 就會開始工作,那速度是一定會相當慢的。這里嘗試舉個不是很恰當?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 對象本身進行操作,而不是生成新的對象,再改變對象引用。所以在一般情況下我們推薦使用 StringBuffer ,特別是字符串對象經(jīng)常改變的情況下。而在某些特別情況下, 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”);

           你會很驚訝的發(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”; 所以當然不需要太多的時間了。但大家這里要注意的是,如果你的字符串是來自另外的 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ū)安全地用于多個線程。可以在必要時對這些方法進行同步,因此任意特定實例上的所有操作就好像是以串行順序發(fā)生的,該順序與所涉及的每個線程進行的方法調(diào)用順序一致。

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

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

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

    在大部分情況下 StringBuilder > StringBuffer

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

    StringBuilder > StringBuffer > String

    既然有這樣的推導結(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 直接進行字符串拼接的測試

        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)該不會當機吧?):

    操作 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 的了。

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

    posted on 2006-05-20 18:34 liaojiyong 閱讀(952) 評論(1)  編輯  收藏 所屬分類: Java

    評論

    # re: String,StringBuffer,StringBuilder(轉(zhuǎn)載) 2006-05-20 19:52 liaojiyong

    經(jīng)典


      回復  更多評論   

    主站蜘蛛池模板: 国产免费毛不卡片| 亚洲精品资源在线| 亚洲午夜国产精品无码| 国产亚洲中文日本不卡二区| 大片免费观看92在线视频线视频 | 男女超爽视频免费播放| 精品成人免费自拍视频| 国产精品亚洲αv天堂无码| 亚洲色图校园春色| 暖暖免费在线中文日本| 国产又黄又爽又猛的免费视频播放 | 国产亚洲精品美女2020久久| 97无码免费人妻超级碰碰夜夜 | 亚洲色欲或者高潮影院| 国产午夜无码精品免费看| 美腿丝袜亚洲综合| 无遮挡免费一区二区三区| 免费v片在线观看品善网| 亚洲精品国产手机| 免费A级毛片无码专区| 久久精品亚洲一区二区| 色屁屁www影院免费观看视频| 中文字幕免费在线看线人| 337p日本欧洲亚洲大胆裸体艺术| 边摸边吃奶边做爽免费视频网站 | **毛片免费观看久久精品| 伊人久久综在合线亚洲2019| 免费在线观看自拍性爱视频| 国产jizzjizz免费看jizz| 亚洲香蕉久久一区二区| 国产免费不卡v片在线观看| 亚洲AV无码一区二区三区在线| 69成人免费视频| 中中文字幕亚洲无线码| 国产精品视频永久免费播放| 亚洲最大无码中文字幕| 69av免费观看| 91亚洲国产成人久久精品网址 | 黄页网站在线看免费| 久久久亚洲欧洲日产国码农村| 精品一区二区三区免费毛片|