成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Java字符串拼接效率分析及更優實踐

開發 后端
java連接字符串有多種方式,比如+操作符,StringBuilder.append方法,這些方法各有什么優劣(可以適當說明各種方式的實現細節)?

[[173287]]

本文來源于問題 Java字符串連接***實踐?

  1. java連接字符串有多種方式,比如+操作符,StringBuilder.append方法,這些方法各有什么優劣(可以適當說明各種方式的實現細節)?
  2. 按照高效的原則,那么java中字符串連接的***實踐是什么?
  3. 有關字符串處理,都有哪些其他的***實踐?

廢話不多說,直接開始, 環境如下:

  • JDK版本: 1.8.0_65
  • CPU: i7 4790
  • 內存: 16G

直接使用+拼接

看下面的代碼:

  1. @Test 
  2.     public void test() { 
  3.         String str1 = "abc"
  4.         String str2 = "def"
  5.         logger.debug(str1 + str2); 
  6.     }  

在上面的代碼中,我們使用加號來連接四個字符串,這種字符串拼接的方式優點很明顯: 代碼簡單直觀,但是對比StringBuilder和StringBuffer在大部分情況下比后者都低,這里說是大部分情況下,我們用javap工具對上面代碼生成的字節碼進行反編譯看看在編譯器對這段代碼做了什么。

  1. public void test(); 
  2.     Code: 
  3.        0: ldc           #5                  // String abc 
  4.        2: astore_1 
  5.        3: ldc           #6                  // String def 
  6.        5: astore_2 
  7.        6: aload_0 
  8.        7: getfield      #4                  // Field logger:Lorg/slf4j/Logger; 
  9.       10: new           #7                  // class java/lang/StringBuilder 
  10.       13: dup 
  11.       14: invokespecial #8                  // Method java/lang/StringBuilder."<init>":()V 
  12.       17: aload_1 
  13.       18: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 
  14.       21: aload_2 
  15.       22: invokevirtual #9                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 
  16.       25: invokevirtual #10                 // Method java/lang/StringBuilder.toString:()Ljava/lang/String; 
  17.       28: invokeinterface #11,  2           // InterfaceMethod org/slf4j/Logger.debug:(Ljava/lang/String;)V 
  18.       33: return  

從反編譯的結果來看,實際上對字符串使用+操作符進行拼接,編譯器會在編譯階段把代碼優化成使用StringBuilder類,并調用append方法進行字符串拼接,***調用toString方法,這樣看來是否可以認為在一般情況下其實直接使用+,反正編譯器也會幫我優化為使用StringBuilder?

StringBuilder源碼分析

答案自然是不可以的,原因就在于StringBuilder這個類它內部做了些什么時。

我們看一看StringBuilder類的構造器

  1. public StringBuilder() { 
  2.         super(16); 
  3.     } 
  4.  
  5.     public StringBuilder(int capacity) { 
  6.         super(capacity); 
  7.     } 
  8.  
  9.     public StringBuilder(String str) { 
  10.         super(str.length() + 16); 
  11.         append(str); 
  12.     } 
  13.  
  14.     public StringBuilder(CharSequence seq) { 
  15.         this(seq.length() + 16); 
  16.         append(seq); 
  17.     }  

StringBuilder提供了4個默認的構造器, 除了無參構造函數外,還提供了另外3個重載版本,而內部都調用父類的super(int capacity)構造方法,它的父類是AbstractStringBuilder,構造方法如下:

  1. AbstractStringBuilder(int capacity) { 
  2.         value = new char[capacity]; 
  3.     }  

可以看到實際上StringBuilder內部使用的是char數組來存儲數據(String、StringBuffer也是),這里capacity的值指定了數組的大小。結合StringBuilder的無參構造函數,可以知道默認的大小是16個字符。

也就是說如果待拼接的字符串總長度不小于16的字符的話,那么其實直接拼接和我們手動寫StringBuilder區別不大,但是我們自己構造StringBuilder類可以指定數組的大小,避免分配過多的內存。

現在我們再看看StringBuilder.append方法內部做了什么事:

  1. @Override 
  2.    public StringBuilder append(String str) { 
  3.        super.append(str); 
  4.        return this; 
  5.    }  

直接調用的父類的append方法:

  1. public AbstractStringBuilder append(String str) { 
  2.         if (str == null
  3.             return appendNull(); 
  4.         int len = str.length(); 
  5.         ensureCapacityInternal(count + len); 
  6.         str.getChars(0, len, value, count); 
  7.         count += len; 
  8.         return this; 
  9.     }  

在這個方法內部調用了ensureCapacityInternal方法,當拼接后的字符串總大小大于內部數組value的大小時,就必須先擴容才能拼接,擴容的代碼如下:

  1. void expandCapacity(int minimumCapacity) { 
  2.         int newCapacity = value.length * 2 + 2; 
  3.         if (newCapacity - minimumCapacity < 0) 
  4.             newCapacity = minimumCapacity; 
  5.         if (newCapacity < 0) { 
  6.             if (minimumCapacity < 0) // overflow 
  7.                 throw new OutOfMemoryError(); 
  8.             newCapacity = Integer.MAX_VALUE; 
  9.         } 
  10.         value = Arrays.copyOf(value, newCapacity); 
  11.     }  

StringBuilder在擴容時把容量增大到當前容量的兩倍+2,這是很可怕的,如果在構造的時候沒有指定容量,那么很有可能在擴容之后占用了浪費大量的內存空間。其次擴容后還調用了Arrays.copyOf方法,這個方法把擴容前的數據復制到擴容后的空間內,這樣做的原因是:StringBuilder內部使用char數組存放數據,java的數組是不可擴容的,所以只能重新申請一片內存空間,并把已有的數據復制到新的空間去,這里它最終調用了System.arraycopy方法來復制,這是一個native方法,底層直接操作內存,所以比我們用循環來復制要塊的多,即便如此,大量申請內存空間和復制數據帶來的影響也不可忽視。

使用+拼接和使用StringBuilder比較

  1. @Test 
  2. public void test() { 
  3.     String str = ""
  4.     for (int i = 0; i < 10000; i++) { 
  5.         str += "asjdkla"
  6.     } 
  7.  

上面這段代碼經過優化后相當于:

  1. @Test 
  2.    public void test() { 
  3.        String str = null
  4.        for (int i = 0; i < 10000; i++) { 
  5.            str = new StringBuilder().append(str).append("asjdkla").toString(); 
  6.        } 
  7.    } 

一眼就能看出創建了太多的StringBuilder對象,而且在每次循環過后str越來越大,導致每次申請的內存空間越來越大,并且當str長度大于16時,每次都要擴容兩次!而實際上toString方法在創建String對象時,調用了Arrays.copyOfRange方法來復制數據,此時相當于每執行一次,擴容了兩次,復制了3次數據,這樣的代價是相當高的。

  1. public void test() { 
  2.         StringBuilder sb = new StringBuilder("asjdkla".length() * 10000); 
  3.         for (int i = 0; i < 10000; i++) { 
  4.             sb.append("asjdkla"); 
  5.         } 
  6.         String str = sb.toString(); 
  7.     }  

這段代碼的執行時間在我的機器上都是0ms(小于1ms)和1ms,而上面那段代碼則大約在380ms!效率的差距相當明顯。

同樣是上面的代碼,將循環次數調整為1000000時,在我的機器上,有指定capacity時耗時大約20ms,沒有指定capacity時耗時大約29ms,這個差距雖然和直接使用+操作符有了很大的提升(且循環次數增大了100倍),但是它依舊會觸發多次擴容和復制。

將上面的代碼改成使用StringBuffer,在我的機器上,耗時大約為33ms,這是因為StringBuffer在大部分方法上都加上了synchronized關鍵字來保證線程安全,執行效率有一定程度上的降低。

使用String.concat拼接

現在再看這段代碼:

  1. @Test 
  2.    public void test() { 
  3.        String str = ""
  4.        for (int i = 0; i < 10000; i++) { 
  5.            str.concat("asjdkla"); 
  6.        } 
  7.    }  

這段代碼使用了String.concat方法,在我的機器上,執行時間大約為130ms,雖然直接相加要好的多,但是比起使用StringBuilder還要太多了,似乎沒什么用。其實并不是,在很多時候,我們只需要連接兩個字符串,而不是多個字符串的拼接,這個時候使用String.concat方法比StringBuilder要簡潔且效率要高。

  1. public String concat(String str) { 
  2.         int otherLen = str.length(); 
  3.         if (otherLen == 0) { 
  4.             return this; 
  5.         } 
  6.         int len = value.length; 
  7.         char buf[] = Arrays.copyOf(value, len + otherLen); 
  8.         str.getChars(buf, len); 
  9.         return new String(buf, true); 
  10.     } 

 上面這段是String.concat的源碼,在這個方法中,調用了一次Arrays.copyOf,并且指定了len + otherLen,相當于分配了一次內存空間,并分別從str1和str2各復制一次數據。而如果使用StringBuilder并指定capacity,相當于分配一次內存空間,并分別從str1和str2各復制一次數據,***因為調用了toString方法,又復制了一次數據。

結論

現在根據上面的分析和測試可以知道:

  1. Java中字符串拼接不要直接使用+拼接。
  2. 使用StringBuilder或者StringBuffer時,盡可能準確地估算capacity,并在構造時指定,避免內存浪費和頻繁的擴容及復制。
  3. 在沒有線程安全問題時使用StringBuilder, 否則使用StringBuffer。
  4. 兩個字符串拼接直接調用String.concat性能***。

關于String的其他***實踐

  1. 用equals時總是把能確定不為空的變量寫在左邊,如使用"".equals(str)判斷空串,避免空指針異常。
  2. 第二點是用來排擠***點的.. 使用str != null && str.length() != 0來判斷空串,效率比***點高。
  3. 在需要把其他對象轉換為字符串對象時,使用String.valueOf(obj)而不是直接調用obj.toString()方法,因為前者已經對空值進行檢測了,不會拋出空指針異常。
  4. 使用String.format()方法對字符串進行格式化輸出。
  5. 在JDK 7及以上版本,可以在switch結構中使用字符串了,所以對于較多的比較,使用switch代替if-else。

我暫時想的起來的就這么幾個了.. 請大家幫忙補充補充...

責任編輯:龐桂玉 來源: segmentfault
相關推薦

2011-07-11 15:36:44

JavaScript

2013-06-24 15:16:29

Java字符串拼接

2021-06-11 18:08:00

Java字符串拼接

2023-11-06 09:32:52

Java實踐

2023-10-31 18:57:02

Java字符串

2021-05-31 07:57:00

拼接字符串Java

2011-07-11 16:00:22

字符串拼接

2022-11-25 07:53:26

bash腳本字符串

2019-02-27 09:08:20

Java 8StringJoineIDEA

2019-12-25 15:41:50

JavaScript程序員編程語言

2021-10-31 23:01:50

語言拼接字符串

2024-12-23 07:38:20

2010-10-09 11:43:10

MYSQL字符串

2021-12-10 08:17:48

字符串拼接場景

2017-01-19 11:26:55

Java 8StringBuild

2025-01-03 08:31:43

2023-12-11 08:39:14

Go語言字符串拼

2016-12-27 09:46:55

Java 8StringBuild

2024-12-20 12:10:19

2021-01-14 05:13:34

倒排索引搜索
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91视视频在线观看入口直接观看 | 欧美一区精品 | 欧美国产一区二区 | 欧美xxxⅹ性欧美大片 | 亚洲一区久久 | 成人欧美日韩一区二区三区 | 91精品91久久久 | 亚洲社区在线 | www.日本国产| 国产欧美一区二区三区在线播放 | 天天干天天爱天天爽 | 天堂成人国产精品一区 | 成人三级在线观看 | 日本黄色不卡视频 | 国产精品成人在线播放 | 超碰成人免费 | 精品99爱视频在线观看 | 一区不卡在线观看 | 在线国产一区 | 国产精品一区在线观看你懂的 | 成人免费视频 | 国产精品一卡二卡三卡 | 一级做a爰片性色毛片16美国 | 精品一区二区三区在线观看 | 一级黄色大片 | 欧美日韩电影一区二区 | 欧洲视频一区二区 | 亚洲成人黄色 | 毛片网站在线观看视频 | 在线免费观看毛片 | 玖草资源 | 婷婷91| 国产乱码精品1区2区3区 | 欧美黄视频 | 激情毛片| 岛国av一区二区三区 | 免费在线观看成年人视频 | 午夜视频在线视频 | 亚洲一区二区电影网 | 99国产精品久久久久 | 国产精品99久久久久久久久久久久 |