追了多年的開發框架,你還認識指針嗎?

来源:https://www.cnblogs.com/huangxincheng/archive/2020/05/16/12898905.html
-Advertisement-
Play Games

一:背景 1. 講故事 高級語言玩多了,可能很多人對指針或者彙編都淡忘了,本篇就和大家聊一聊指針,雖然C 中是不提倡使用的,但你能說指針在C 中不重要嗎?你要知道FCL內庫中大量的使用指針,如 等等數不勝數,如例代碼: 對,你覺得的美好世界,其實都是別人幫你負重前行,退一步說,指針的理解和不理解,對 ...


一:背景

1. 講故事

高級語言玩多了,可能很多人對指針或者彙編都淡忘了,本篇就和大家聊一聊指針,雖然C#中是不提倡使用的,但你能說指針在C#中不重要嗎?你要知道FCL內庫中大量的使用指針,如String,Encoding,FileStream等等數不勝數,如例代碼:


	private unsafe static bool EqualsHelper(string strA, string strB)
	{
		fixed (char* ptr = &strA.m_firstChar)
		{
			fixed (char* ptr3 = &strB.m_firstChar)
			{
				char* ptr2 = ptr;
				char* ptr4 = ptr3;
				while (num >= 12) {...}
				while (num > 0 && *(int*)ptr2 == *(int*)ptr4) {...}
			}
		}
	}

	public unsafe Mutex(bool initiallyOwned, string name, out bool createdNew, MutexSecurity mutexSecurity)
	{
		byte* ptr = stackalloc byte[(int)checked(unchecked((ulong)(uint)securityDescriptorBinaryForm.Length))]
	}
   
    private unsafe int ReadFileNative(SafeFileHandle handle, byte[] bytes, out int hr)
    {
        fixed (byte* ptr = bytes)
		{
			num = ((!_isAsync) ? Win32Native.ReadFile(handle, ptr + offset, count, out numBytesRead, IntPtr.Zero) : Win32Native.ReadFile(handle, ptr + offset, count, IntPtr.Zero, overlapped));
		}
   }    

對,你覺得的美好世界,其實都是別人幫你負重前行,退一步說,指針的理解和不理解,對你研究底層源碼影響是不能忽視的,指針相對比較抽象,考的是你的空間想象能力,可能現存的不少程式員還是不太明白,因為你缺乏所見即所得的工具,希望這一篇能幫你少走些彎路。

二:windbg助你理解

指針雖然比較抽象,但如果用windbg實時查看記憶體佈局,就很容易幫你理解指針的套路,下麵先理解下指針的一些簡單概念。

1. &、* 運算符

&取址運算符,用於獲取某一個變數的記憶體地址, *運算符,用於獲取指針變數中存儲地址指向的值,很抽象吧,看windbg。

            unsafe
            {
                int num = 10;
                int* ptr = #
                var num2 = *ptr;
                Console.WriteLine(num2);
            }

0:000> !clrstack -l
OS Thread Id: 0x41ec (0)
        Child SP               IP Call Site
0000005b1efff040 00007ffc766208e2 *** WARNING: Unable to verify checksum for ConsoleApp4.exe
ConsoleApp4.Program.Main(System.String[]) [C:\dream\Csharp\ConsoleApp1\ConsoleApp4\Program.cs @ 25]
    LOCALS:
        0x0000005b1efff084 = 0x000000000000000a
        0x0000005b1efff078 = 0x0000005b1efff084
        0x0000005b1efff074 = 0x000000000000000a

仔細觀察 LOCALS 中三組鍵值對。

<1> int* ptr = &num; => 0x0000005b1efff078 = 0x0000005b1efff084

int* ptr叫做指針變數,既然是變數必須得有自己的棧上地址 0x0000005b1efff078 ,而這個地址上的值為 0x0000005b1efff084,這不就是num的棧地址嘛,嘿嘿。

<2> var num2 = *ptr; => 0x0000005b1efff074 = 0x000000000000000a

*ptr 就是用ptr的value [0x0000005b1efff084] 獲取這個地址指向的值,所以就是10啦。

如果不明白,我畫一張圖,這可是重中之重哦~

2. **運算符

** 也叫二級指針,指向一級指針變數地址的指針,有點意思,如下程式:ptr2指向的就是 ptr的棧上地址, 一圖勝千言。


    unsafe
    {
        int num1 = 10;
        int* ptr = &num1;
        int** ptr2 = &ptr;
        var num2 = **ptr2;
    }


0:000> !clrstack -l
ConsoleApp4.Program.Main(System.String[]) [C:\dream\Csharp\ConsoleApp1\ConsoleApp4\Program.cs @ 26]
    LOCALS:
        0x000000305f5fef24 = 0x000000000000000a
        0x000000305f5fef18 = 0x000000305f5fef24
        0x000000305f5fef10 = 0x000000305f5fef18
        0x000000305f5fef0c = 0x000000000000000a

3. ++、--運算符

這種算術操作常常用在數組或者字元串等值類型集合,比如下麵代碼:

    fixed (int* ptr = new int[3] { 1, 2, 3 }) { }
    fixed (char* ptr2 = "abcd") { }

首先ptr預設指向數組在堆上分配的首地址,也就是1的記憶體地址,當ptr++後會進入到下一個整形元素2的記憶體地址,再++後又進入下一個int的記憶體地址,也就是3,很簡單吧,我舉一個例子:

        unsafe
        {
            fixed (int* ptr = new int[3] { 1, 2, 3 })
            {
                int* cptr = ptr;
			    Console.WriteLine(((long)cptr++).ToString("x16"));
				Console.WriteLine(((long)cptr++).ToString("x16"));
				Console.WriteLine(((long)cptr++).ToString("x16"));
            }
        }

0:000> !clrstack -l
    LOCALS:
        0x00000070c15fea50 = 0x000001bcaac82da0
        0x00000070c15fea48 = 0x0000000000000000
        0x00000070c15fea40 = 0x000001bcaac82dac
        0x00000070c15fea38 = 0x000001bcaac82da8

一圖勝千言哈,Console中的三個記憶體地址分別存的值是1,2,3哈, 不過這裡要註意的是,C#是托管語言,引用類型是分配在托管堆中,所以堆上地址會存在變動的可能性,這是因為GC會定期回收記憶體,所以vs編譯器需要你用fixed把堆上記憶體地址固定住來逃過GC的打壓,在本例中就是 0x000001bcaac82da0 - (0x000001bcaac82da8 +4)

三:用兩個案例幫你理解

古語說的好,一言不中,千言無用,你得拿一些例子活講活用,好吧,準備兩個例子。

1. 使用指針對string中的字元進行替換

我們都知道string中有一個replace方法,用於將指定的字元替換成你想要的字元,可是C#中的string是不可變的,你就是對它吐口痰它都會生成一個新字元串,

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

-Advertisement-
Play Games
更多相關文章
  • 大家好,520它又要來了 所以今天的主題是粉色的 為了各位禿頭程式員不再頭疼 本文給大家介紹幾種用Python表白的姿勢 絕不是畫個愛心曲線那麼簡單~ 、 ​ 屬於TA的詞雲 用Python將你們的 聊天記錄/TA的朋友圈文字 製作成漂亮的詞雲圖,先來看看效果 ​ 當然圖片你可以隨便選擇,愛心、玫瑰 ...
  • (一)數組 數組(Array)是一種線性表數據結構。它用一組連續的記憶體空間,來存儲一組具有相同類型的數據。 1、數組支持隨機訪問,根據下標隨機訪問的時間複雜度為 O(1)。 通過 a[i]_address = a[0]_address + i*元素的大小(位元組) ,得到a[i]所在的位置。 2、插入 ...
  • 前言 本文的文字及圖片來源於網路,僅供學習、交流使用,不具有任何商業用途,版權歸原作者所有,如有問題請及時聯繫我們以作處理。 大學畢業準備實習和工作的時大家在實習和工作的時候,我們都不可避免地要面對租房的問題,尤其是想去一線城市工作。“用數據說話”這句話我們肯定再熟悉不過了,能用數據進行客觀分析的, ...
  • 學習網址:https://docs.microsoft.com/zh-cn/dotnet/standard/exceptions/exception-class-and-properties 瞭解Exception的常見屬性。 Data:Exception的鍵/值對數據。 using System; ...
  • 官網 http://www.hzhcontrols.com/ 前提 入行已經7,8年了,一直想做一套漂亮點的自定義控制項,於是就有了本系列文章。 GitHub:https://github.com/kwwwvagaa/NetWinformControl 碼雲:https://gitee.com/kww ...
  • ​分享一套基於微信公眾號介面的O2O購物商城源碼;有興趣下載源碼學習的朋友可以付費閱讀後,在文尾查看下載鏈接。 此商城的特點如下: 1、使用ASP.NET MVC4 開發 2、Amaze UI & WeUI & Jquery 作為前端框架 3、使用Senparc.Weixin.MP 調用微信支付等接 ...
  • 普通商戶分賬功能 分賬比例:目前只有”低比例分賬“小於等於30%分賬,分賬金額需要減去(千6)手續費. 每一張訂單隻能分發,當前訂單總額的百分之30可以分賬; 比如:一張訂單支付金額100元,這張訂單隻能分發29.82元。 (100-100*0.006)*30% 的金額 開發文檔地址:https:/ ...
  • 程式執行時可能會遇到一些無法預料到的錯誤,這稱作異常。.Net中的異常都繼承System.Exception。異常會在堆棧中往上傳遞,直到被處理或者程式終止。 常見異常: IndexOutOfRangeException 數組索引超出使用範圍,運行時引發。 NullReferenceExceptio ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...