自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

為什么Java 8中不再需要StringBuilder拼接字符串

開發(fā) 后端
在Java中,字符串對(duì)象是不可變的,意思是它一旦創(chuàng)建,你就無法再改變它。所以在我們拼接字符串的時(shí)候,創(chuàng)建了一個(gè)新的字符串,舊的被垃圾回收器所標(biāo)記。

在Java開發(fā)者中,字符串的拼接占用資源高往往是熱議的話題.

讓我們深入討論一下為什么會(huì)占用高資源。

在Java中,字符串對(duì)象是不可變的,意思是它一旦創(chuàng)建,你就無法再改變它。所以在我們拼接字符串的時(shí)候,創(chuàng)建了一個(gè)新的字符串,舊的被垃圾回收器所標(biāo)記。

如果我們處理上百萬的字符串,然后,我們就會(huì)生成百萬的額外字符串被垃圾回收器處理。

虛擬機(jī)底層在拼接字符串時(shí)執(zhí)行了眾多操作。拼接字符串最直接的點(diǎn)操作(dot operator)就是String#concat(String)操作。

 

  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.  
  11. public static char[] copyOf(char[] original, int newLength) { 
  12.     char[] copy = new char[newLength]; 
  13.     System.arraycopy(original, 0, copy, 0
  14.                      Math.min(original.length, newLength)); 
  15.     return copy; 
  16.  
  17. void getChars(char dst[], int dstBegin) { 
  18.     System.arraycopy(value, 0, dst, dstBegin, value.length); 

你可以看到一個(gè)字符數(shù)組被創(chuàng)建,長(zhǎng)度則是已有字符和拼接的字符長(zhǎng)度之和。然后,它們的值復(fù)制到新的字符數(shù)組中。***,用這個(gè)字符數(shù)組創(chuàng)建一個(gè)String對(duì)象并返回。

所以這些操作繁多,如果你計(jì)算一下,會(huì)發(fā)現(xiàn)是O(n^2)的復(fù)雜度。

為了解決這個(gè)問題,我們使用StringBuilder類。它就像可變的String類。拼接方法幫助我們避免不必要的復(fù)制。它擁有O(n)的復(fù)雜度,遠(yuǎn)遠(yuǎn)優(yōu)于O(n^2)。

然而Java 8默認(rèn)使用StringBuilder拼接字符串。

Java 8的文檔說明:

為了提高字字符串拼接的性能,Java編譯器可以使用StringBuffer類或類似技術(shù),在使用求值表達(dá)式時(shí),減少中間String對(duì)象的創(chuàng)建。

Java編譯器處理這種情況:

  1. public class StringConcatenateDemo { 
  2.   public static void main(String[] args) { 
  3.      String str = "Hello "; str += "world"
  4.    } 

上面的代碼會(huì)被編譯成如下字節(jié)碼:

 

  1. public class StringConcatenateDemo { 
  2.   public StringConcatenateDemo(); 
  3.     Code: 
  4.        0: aload_0 
  5.        1: invokespecial #1                  // Method java/lang/Object."<init>":()V 
  6.        4return 
  7.   public static void main(java.lang.String[]); 
  8.     Code: 
  9.        0: ldc           #2                  // String Hello 
  10.        2: astore_1 
  11.        3new           #3                  // class java/lang/StringBuilder 
  12.        6: dup 
  13.        7: invokespecial #4                  // Method java/lang/StringBuilder."<init>":()V 
  14.       10: aload_1 
  15.       11: invokevirtual #5                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 
  16.       14: ldc           #6                  // String world 
  17.       16: invokevirtual #5                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 
  18.       19: invokevirtual #7                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String; 
  19.       22: astore_1 
  20.       23return 

你可以在這些字節(jié)碼中看到,使用了StringBuilder。所以我們?cè)贘ava 8中不再需要使用StringBuilder類。

譯文鏈接:http://www.codeceo.com/article/why-java8-not-use-stringbuilder.html
英文原文:We Don't Need StringBuilder for Concatenation Anymore
翻譯作者:碼農(nóng)網(wǎng) – 孫騰浩

責(zé)任編輯:王雪燕 來源: 碼農(nóng)網(wǎng)
相關(guān)推薦

2017-01-19 11:26:55

Java 8StringBuild

2025-02-27 08:18:40

2019-02-27 09:08:20

Java 8StringJoineIDEA

2021-04-21 10:36:47

StringBuildJava8StringJoine

2024-08-16 22:06:06

2011-07-11 16:00:22

字符串拼接

2022-11-25 07:53:26

bash腳本字符串

2013-06-24 15:16:29

Java字符串拼接

2021-06-11 18:08:00

Java字符串拼接

2015-03-19 15:04:06

2021-01-07 09:20:08

SQL字符串Server

2023-10-31 18:57:02

Java字符串

2025-01-03 08:31:43

2021-05-31 07:57:00

拼接字符串Java

2023-01-03 08:07:33

Go字符串指針

2016-10-12 10:18:53

Java字符串源碼分析

2019-01-29 10:30:32

阿里巴巴Java字符串

2019-02-27 09:00:13

阿里巴巴for循環(huán)Java

2019-12-25 15:41:50

JavaScript程序員編程語言

2021-10-31 23:01:50

語言拼接字符串
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)