Java String:重要到別人只能當老二的字元串類

来源:https://www.cnblogs.com/qing-gee/archive/2019/01/18/10277688.html
-Advertisement-
Play Games

字元串,是Java中最重要的類。這句肯定的推斷不是Java之父詹姆斯·高斯林說的,而是沉默王二說的,因此你不必懷疑它的準確性。 關於字元串,有很多的面試題,但我總覺得理論知識繞來繞去沒多大意思。你比如說:String cmower = new String("沉默王二");定義了幾個對象? 我總覺得 ...


字元串,是Java中最重要的類。這句肯定的推斷不是Java之父詹姆斯·高斯林說的,而是沉默王二說的,因此你不必懷疑它的準確性。

關於字元串,有很多的面試題,但我總覺得理論知識繞來繞去沒多大意思。你比如說:String cmower = new String("沉默王二");定義了幾個對象?

我總覺得問我這樣的問題,就好像是在拷問我:“既然你家買了冰箱,你難道不應該知道冰箱製冷的原理?”

再說,為什麼要用String cmower = new String("沉默王二");而不是String cmower = "沉默王二";

我勸各位面試官不要再纏住這樣的問題不放了,切記“學以致用”。理論知識如果一直是在繞彎彎,那真的毫無價值。如果要我來做面試官,我想要問的問題是:“你平常是怎麼判斷兩個字元串相等的?是用equals()還是==?”

前言就說這麼多。接下來,我們來探討幾個實用的知識點。

01 字元串是不可變的

我們來看一下String類的定義:

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
}

可以發現,String類是final類型的,因此不能被繼承。

如果類可以被繼承,那麼就會破壞類的不可變性機制。因為子類可以覆蓋父類的方法,並且可以改變父類的成員變數值,一旦子類以父類的形式出現時,就不能保證類是不可變的。

String類的不可變性有什麼好處呢?

1)作為HashMap的鍵。

因為字元串是不可變的,因此它在創建的時候哈希碼(hash code)就計算好了。這也就意味著每次在使用一個字元串的哈希碼的時候不用重新計算一次,這樣更加高效,很適合作為HashMap中的鍵。

2)線程安全。

同一個字元串對象可以被多個線程共用,如果訪問頻繁的話,可以省略同步和鎖等待的時間,從而提升性能。

3)字元串常量池的需要。

稍後介紹。

特別要註意的是,String類的所有方法都沒有改變字元串本身的值,都是返回了一個新的對象

02 字元串常量池

在Java中,常用的創建字元串的方式有兩種:

String cmower = "沉默王二";

String cmowsan = new String("沉默王三");

cmower使用雙引號,cmowsan使用new關鍵字,它們有什麼區別呢?

答案如下:

String cmower = "沉默王二";
String cmower1 = "沉默王二";
System.out.println(cmower == cmower1); // 輸出true

String cmowsan = new String("沉默王三");
String cmowsan1 = new String("沉默王三");
System.out.println(cmowsan == cmowsan1); // 輸出false

雙引號創建的相同字元串使用==判斷時結果為true,而new關鍵字創建的相同字元串使用==判斷時結果為false。

這是為什麼呢?

String在Java中使用過於頻繁,為了避免在系統中產生大量的String對象,Java的設計者引入了“字元串常量池”的概念

當使用雙引號創建一個字元串時,首先會檢查字元串常量池中是否有相同的字元串對象,如果有,則直接從常量池中取出對象引用;如果沒有,則新建字元串對象,並將其放入字元串常量池中,並返回對象引用。

這也就是說,"沉默王二"是放在字元串常量池中的,cmower和cmower1兩個字元串對象引用是相同的。

而new關鍵字創建的字元串對象是不涉及字元串常量池的,直接放在堆中,也就是說,雖然cmowsan和cmowsan1都叫沉默王三,但不一個人。

強烈建議:不要使用new關鍵字的形式創建字元串對象。

03 +號和StringBuilder

由於字元串是不可變的,因此字元串在進行拼接的時候會創建新的字元串對象。大家都知道,記憶體是一定的,因此對象創建多了就會影響系統性能。

StringBuilder正是為瞭解決字元串拼接產生太多中間對象的問題而提供的一個類,可以通過append()方法把字元串添加到已有序列的末尾,非常高效。

那麼有人在進行字元串拼接的時候,就會產生疑惑:“我到底是用+號還是StringBuilder?”

我們先來看這樣一段代碼:

String chenmo = "沉默";
String wanger = "王二";
System.out.println(chenmo + wanger);

這段代碼是怎麼編譯的呢?可以使用JAD(Java反編譯工具)來看一看。

String s = "\u5A0C\u5910\u7CAF";
String s1 = "\u941C\u5B29\u7C29";
System.out.println((new StringBuilder()).append(s).append(s1).toString());

你是不是看到了StringBuilder的影子?

沒錯,使用+號進行字元串拼接的時候,Java編譯器實際是通過StringBuilder類來完成的。

難道可以使用+號來隨意拼接字元串?反正Java編譯器已經自動地為我們優化了。

但事實並非如此,來看這樣一段代碼:

String cmowers = "";
for (int i = 0; i < 9; i++) {
    cmowers += "沉默王二";
}
System.out.println(cmowers);

閉上眼睛先想一想,Java編譯器會怎麼做?我們期望的結果是在迴圈外部就創建StringBuilder,Java編譯器能如我們所願嗎?

JAD反編譯後的結果如下:

String s = "";
for(int i = 0; i < 10; i++)
    s = (new StringBuilder()).append(s).append("\u5A0C\u5910\u7CAF\u941C\u5B29\u7C29").toString();

System.out.println(s);

這麼看來,StringBuilder是在for迴圈內部創建的,也就是說會創建10次。天吶,這可不是我們期望的結果!我們只希望StringBuilder創建一次。

沒辦法,Java編譯器是做不到的,只能靠我們自己:

StringBuilder cmowers = new StringBuilder();
for (int i = 0; i < 9; i++) {
    cmowers.append("沉默王二");
}
System.out.println(cmowers);

強烈建議:如果只是三四個字元串的拼接,儘管使用+號操作符,別想什麼性能優化(舉個例子,你離目的地只有100米,你是打算打個計程車,還是自己步行走過去?);如果遇到多於四個字元串的拼接,或者需要用到迴圈來拼接,那就選擇StringBuilder。

在我年輕的時候,我還會犯這樣一個錯誤:

StringBuilder cmowers = new StringBuilder();
for (int i = 0; i < 9; i++) {
    cmowers.append("沉默王二" + "和他的讀者朋友們");
}
System.out.println(cmowers);

我去,竟然在append()方法的內部使用+號!因為這個錯誤,我差點沒被領導打死。你可要小心點。

04 關於concat()

除了使用+號和StringBuilder對字元串進行拼接,還可以使用String類的concat()方法。

concat()方法只不過是String類的一個方法而已,為什麼我要單獨拎出來說呢?

因為之前我要在JSP頁面的EL表達式中拼接字元串,剛開始想到的是用+號操作符,但EL表達式不是Java,+號操作符是不能拼接字元串的。我當時竟然沒想起來用concat()

重新銘記一下:

${item.username.concat('-').concat(item.realname)}

05 關於intern()

關於字元串的性能問題,我常在一些技術文章中看到這樣的建議:“如果一個字元串使用的頻率非常高,建議使用String.intern()將其緩存。”

但我並不建議你這麼做,因為這個方法要顯式的調用,這樣很麻煩;況且,在代碼編寫階段,怎麼可能知道哪個字元串使用頻率很高呢?

06 關於StringUtils

據我的編程經驗來看,字元串的操作往往需要用到一個工具類,那就是org.apache.commons.lang3.StringUtils(null安全的,也就是說,StringUtils類的方法可以接受為null的字元串,但不會拋出NullPointerException)。

不過,我最常用的方法就那麼幾個:

方法等價
IsEmpty(String str) str == null || str.length == 0
isBlank(String str) str == null || str.length == 0 || str.trim().length == 0
join(Object[] arrey) 把數組中的元素連接成一個字元串返回

 

 

推薦閱讀:

Java異常處理:給程式罩一層保險
Java集合類:我其實沒那麼簡單


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 使用Jquery實現tab鍵切換,代碼簡潔易懂,實現邏輯清晰明瞭。具體總結如下: 需求分析: 滑鼠進入tab切換模塊,滑鼠當前的模塊上邊框變為紅色,並顯示對應的商品名稱。滑鼠離開後,上邊框恢複原色,圖片改變。 實現效果如下: 頁面1: 頁面2 頁面3 頁面4 頁面結構分析: 1.使用大盒子wrapp ...
  • <!DOCTYPE html><html lang="en"><head> <meta charset="utf-8"> <meta http-equiv="x-ua-compatible" content="ie=edge"> <meta name="viewport" content="widt ...
  • 來認識更多的表單吧,增加知識面 我只創建了一個index.html幫助你認識它們 以下是代碼 1 <!DOCTYPE html> 2 <html> 3 <head> 4 <meta charset = "utf-8"> 5 </head> 6 <body> 7 <!--單按鈕輸入(單選)--> 8 ...
  • 先看看表單如何工作吧 請求 響應 簡要工作流程: 瀏覽器載入頁面 用戶輸入數據 用戶提交表單 伺服器響應 概念都清楚了,我們來寫表單吧 只有一個html文件 這是顯示 你可以向空白框框里寫一些東西,然後點擊提交 數據會發到web伺服器的contest.php里,當然了,你也可以自己寫一個php文件 ...
  • let命令的用法 是es6中的聲明一個變數的命令,只在它聲明的代碼塊中有效,出了這個代碼塊就會報錯。也非常適合 迴圈,在迴圈中i的值只在迴圈語句中生效,在外邊取不到的。 命令聲明的是一個全局的變數,i是指向全局的變數,只會輸出最後的值。而 只在迴圈語句塊裡面生效,每次迴圈都會重新聲明一個i的,所以每 ...
  • 如果這是第二次看到我的文章,歡迎右側掃碼訂閱我喲~ 👉 本文長度為2728字,建議閱讀8分鐘。 堅持原創,每一篇都是用心之作~ 前面聊完的2個章節「數據一致性」和「高可用」其實本質是一個通過提升複雜度讓整體更完善的方式。 接下去我們開始聊一些讓系統更簡單,更容易維護的東西——「易伸縮」,首當其衝的 ...
  •   需求變更是信息化過程中的家常便飯,而在變更過程中如何儘可能小的影響線上業務是比較頭疼的事情。舉個車聯網監控的例子:原終端設備上傳車輛的經緯度數據,新的終端設備支持同時上傳速度數據,而舊的車輛狀態表數據量超過億級,此時如果Alter table add column將會造成數據 ...
  • 一、問題描述 Tomcat下麵部署很多個java項目的war包,tomcat啟動一段時間後,發現cpu占用過高,整個界面卡死! 二、通過process explorer查看java進程下的線程 process explorer是Windows系統和應用程式監視工具。 process explorer ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...