字元串 --- 不可變性與駐留池

来源:https://www.cnblogs.com/pandefu/archive/2023/10/18/17771369.html
-Advertisement-
Play Games

引言 面試中,常會問道,在大數據量的字元串拼接情況,為什麼 StringBuilder 性能比直接字元串拼接更好? 主要原因就是 string 是不可變類型,每次操作都會創建新的字元串對象,頻繁操作會導致記憶體頻繁的分配和回收,就會降低性能, 而 StringBuilder 是可變類型,它允許對字元串 ...


引言

面試中,常會問道,在大數據量的字元串拼接情況,為什麼 StringBuilder 性能比直接字元串拼接更好?

主要原因就是 string 是不可變類型,每次操作都會創建新的字元串對象,頻繁操作會導致記憶體頻繁的分配和回收,就會降低性能, 而 StringBuilder 是可變類型,它允許對字元串進行原地修改,無需每次都創建新對象,其內部使用一個緩衝區來存儲字元,可以高效地執行字元串操作,如添加、插入、刪除等。

面試題就不多說了,既然這裡已經提到了字元串性能,那我們來說一說保證字元串的性能、記憶體效率和安全性的兩大門神:

  • 字元串的不可變性
  • 字元串駐留池

原理與關係

C# 中的字元串駐留池(String Interning Pool)是一個關鍵的記憶體管理概念,旨在提高字元串的性能和記憶體效率。字元串駐留池是一個特殊的記憶體區域,用於存儲字元串字面值的唯一實例,以減少記憶體使用和提高性能。

字元串字面值

字元串字面值是指由雙引號括起來的字元序列,比如:"Hello, World!"。字元串字面值通常用於聲明字元串變數或進行字元串操作。這些字元串字面值在編譯時被解析,並根據它們的值存儲在記憶體中。

下麵聲明瞭兩個字元串字面值:

 String s1 = "hello";
 String s2 = "world";

字元串不可變性

字元串不可變,這意味著一旦創建,字元串的內容不能被更改。這種不可變性是為了確保字元串的安全性和可靠性。當你對字元串進行操作時,實際上是創建了新的字元串對象,而原始字元串保持不變。這對於多線程和記憶體管理非常重要。

string originalString = "Hello, World!"; // 創建一個字元串

Console.WriteLine("原始字元串:" + originalString);

// 嘗試修改字元串內容
// 下麵的行將引發編譯錯誤,因為字元串是不可變的
// originalString[0] = 'M';

// 創建新字元串而不是修改原始字元串
string newString = originalString.Replace('H', 'M');
Console.WriteLine("修改後的字元串:" + newString);

Console.WriteLine("原始字元串:" + originalString); // 原始字元串不受影響

Console.WriteLine(object.ReferenceEquals(originalString, newString)); // 不是同一對象

上述代碼輸出:

原始字元串:Hello, World!
修改後的字元串:Mello, World!
原始字元串:Hello, World!
False

字元串駐留池的工作原理

字元串駐留池的核心概念是確保具有相同值的字元串在記憶體中只有一個實例。它的工作原理如下:

  1. 字元串字面值的存儲:當你在代碼中使用字元串字面值時,編譯器會將這些字元串字面值存儲在字元串駐留池中。這是編譯時操作,而不是運行時操作。

  2. 檢查字元串值:在創建字元串字面值時,編譯器會首先檢查字元串池,看是否已經存在具有相同值的字元串。如果存在,編譯器會返回對現有字元串的引用,而不是創建一個新的字元串對象。

  3. 共用相同的實例:如果多個字元串字面值具有相同的值,它們會共用相同的記憶體實例,從而節省記憶體。這意味著即使你多次創建相同值的字元串,實際上它們指向的是相同的記憶體位置。

  4. 不可變性的重要性:字元串的不可變性是字元串駐留池的基礎。因為字元串是不可變的,共用字元串實例不會導致數據損壞或不一致性。

字元串駐留池的優點

字元串駐留池的存在帶來了多個重要優點:

  1. 記憶體節省:由於字元串駐留,相同的字元串值只需存儲一次,減少了記憶體使用。這對於大規模應用程式和處理大量文本數據尤為重要。

  2. 性能提升:由於字元串共用相同的實例,比較字元串的相等性變得更快速,因為可以直接比較引用,而不必比較字元串的內容。

  3. 可靠性:字元串駐留池有助於確保字元串數據的一致性。如果多個部分使用相同的字元串值,它們將引用相同的實例,從而避免數據不一致性。

  4. 簡化代碼:開發人員可以放心地使用字元串字面值,而不必擔心記憶體管理。這使得代碼更簡潔和易於維護。

使用字元串駐留池

通常情況下,你不需要手動管理字元串駐留池,因為C#編譯器和運行時會自動處理字元串的駐留。這意味著當你聲明多個相同值的字元串時,它們將共用相同的記憶體實例,無需任何額外的代碼。

string s1 = "Hello";
string s2 = "Hello";

Console.WriteLine(object.ReferenceEquals(s1, s1)); //輸出True

然而,如果你需要顯式地將一個字元串添加到字元串駐留池中,可以使用string.Intern()方法:

 string s1 = "Hello";
 string s2 = "World";
 // 手動將字元串s2添加到字元串駐留池
 string internedString = string.Intern(s2);
 // 現在s2和internedString都指向相同的字元串對象
 Console.WriteLine(object.ReferenceEquals(s2, internedString));   //輸出True

兩者關係

字元串的不可變性和字元串駐留池之間存在緊密的關係,它們共同作用於C#中的字元串處理和記憶體管理。
這兩個概念之間的關係在以下方面體現:

  • 記憶體共用:由於字元串的不可變性,可以安全地在字元串之間共用記憶體實例。字元串的不可變性確保了多個字元串可以指向相同的記憶體位置,而不必擔心數據被修改。字元串駐留池利用這一點,確保相同值的字元串字面值共用相同的記憶體。

  • 性能和記憶體優化:由於字元串不可變且字元串駐留池的存在,比較字元串的相等性變得更加高效,因為可以直接比較引用而不必比較字元串內容。這提高了字元串操作的性能,同時減少了記憶體使用,因為相同值的字元串只需存儲一次。

  • 共用和復用:字元串不可變性和字元串駐留池的結合使得相同的字元串字面值可以被多個部分共用和復用,從而減少了記憶體開銷。這對於具有重覆字元串值的大型應用程式和處理大量文本數據的情況尤其有益。

總結

綜上所述,字元串的不可變性和字元串駐留池共同提高了C#中字元串的性能、記憶體效率和安全性,使得多個部分可以共用相同值的字元串實例,同時確保字元串的內容不會被無意修改。這些概念在C#中的字元串處理中發揮著關鍵作用。

作者: Niuery Daily

出處: https://www.cnblogs.com/pandefu/>

郵箱: [email protected]

關於作者:.Net Framework,.Net Core ,WindowsForm,WPF ,控制項庫,多線程

本文版權歸作者所有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出 原文鏈接,否則保留追究法律責任的權利。 如有問題, 可郵件咨詢。


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

-Advertisement-
Play Games
更多相關文章
  • 在本文中,我們將全面深入地探討Go語言的反射機制。從反射的基礎概念、為什麼需要反射,到如何在Go中實現反射,以及在高級編程場景如泛型編程和插件架構中的應用,本文為您提供一站式的學習指南。 關註【TechLeadCloud】,分享互聯網架構、雲服務技術的全維度知識。作者擁有10+年互聯網服務架構、AI ...
  • 散點圖,又名點圖、散佈圖、X-Y圖,是將所有的數據以點的形式展現在平面直角坐標繫上的統計圖表。 散點圖常被用於分析變數之間的相關性。如果兩個變數的散點看上去都在一條直線附近波動,則稱變數之間是線性相關的;如果所有點看上去都在某條曲線(非直線)附近波動,則稱此相關為非線形相關的;如果所有點在圖中沒有顯 ...
  • 一、前言 你有看到過那種不間斷型的、迴圈播放視頻音樂的直播間嗎?或者那種直播播放電影的直播間?還有層出不窮的文章,類似如下標題: “如何搭建一個24小時不間斷的直播間?躺入xxxx元!” “24小時電影直播間,每天到賬xxx~xxxx,不出鏡副業,人人可做!” “50塊的雲伺服器直播推流讓我月入過千 ...
  • 一、前言 我們在開發中最常見的異常就是NullPointerException,防不勝防啊,相信大家肯定被坑過! 這種基本出現在獲取資料庫信息中、三方介面,獲取的對象為空,再去get出現! 解決方案當然簡單,只需要判斷一下,不是空在去後續操作,為空返回! 所有在JDK8時出現了專門處理的方案,出來很 ...
  • 作為一為 Java 開發工程師,寫數據的查詢 SQL 是必備的技能。在 日常生活中,是否統計過讀數據和寫數據的頻率。以來開發經驗來說,查詢數據的操作語言是多於寫數據的。 有的信息系統,數據只初始化一次,甚至是服務一輩子。 接觸過很多的 web 開發系統,都是為了管理數據而生存的。要產生數據,才能管理 ...
  • 對於網路通信中的服務端來說,顯然不可能是一對一的,我們所希望的是服務端啟用一份則可以選擇性的與特定一個客戶端通信,而當不需要與客戶端通信時,則只需要將該套接字掛到鏈表中存儲並等待後續操作,套接字服務端通過多線程實現存儲套接字和選擇通信,可以提高服務端的併發性能,使其能夠同時處理多個客戶端的請求。在實... ...
  • MD5的全稱是message-digest algorithm 5,MD5演算法屬於hash演算法的一類,對消息進行運算,產生一個128位的消息摘要(32位數字字母),並且加密不可逆,相同數據的加密結果一定相同。 C# 實現MD5加密。MD5加密16位和32位 ...
  • C# 中的 Queue 是一種基於鏈表的先進先出 (FIFO) 數據結構。以下是一個簡單的 Queue 實例: /// <summary> /// 普通隊列 /// </summary> public void QueueShow() { // 創建一個Queue Queue<string> que ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...