亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C#操作內存的指針怎么理解

發布時間:2021-12-03 09:51:58 來源:億速云 閱讀:148 作者:iii 欄目:編程語言

本篇內容介紹了“C#操作內存的指針怎么理解”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

C#操作內存通過指針之托管代碼

一般來說你在寫任意一個C#程序的時候,你都是在創建托管代碼。托管代碼是在Common Language Runtime (CLR)控制下執行的,CLR使得程序員不需要管理內存和關心內存的分配和回收。CLR也允許你寫非安全代碼 (unsafe code)。

C#操作內存通過指針之非安全代碼

非安全代碼就是不在 CLR 完全控制下執行的代碼,它有可能會導致一些問題,因此他們必須用 “unsafe” 進行表明:

...  unsafe {  ...  // unsafe context: can use pointers here  ...  }  ...

在其他一些地方也可以使用關鍵字 ‘unsafe’,例如我們可以將類或方法表明為非安全的:

unsafe class Class1 {}  static unsafe void FastMove ( int* pi, int* pdi, int length) {...}

‘unsafe’ 關鍵字的必要性是它可以防止程序員的一些意外的用法。你可能會問既然是不安全的為什么還有人要用它。答案就是有時候,在有些情況下,還需要用到指針。

C#操作內存之指針

指針是一種用來存儲其他變量地址的特殊的變量,如果你把***個變量的地址賦給第二個變量,你可以說***個變量是指向第二個,CLR支持3種指針類型:受托管指針, 非托管指針和非托管函數指針。受托管指針存儲在堆上的托管塊的引用,一個非托管指針是傳統的C++指針并且每次使用必須要放在unsafe代碼塊中,一個非托管函數指針也是指向函數地址的傳統的C++指針(delegates 可以被看做是非托管函數指針).

你可以像下面這樣的聲明來創建指針:

類型* 變量_名稱;

既然類型可以是任意一個非引用類型并且不包含引用類型字段,它只能是:sbyte, byte, short, ushort, int, uint, long, ulong, char, float, double, decimal, bool 和枚舉類型以及其他指針類型,也可以是任何用戶自定義的包括非托管類型字段的結構體.

下面是不同類型指針聲明的示例:

int* pi //declaration a pointer to integer variable  float* pf, pq // two pointers to float variables. Not *pf, *pq   char* pz // pointer to char

就像前面說的非托管代碼CLR是不能驗證的,為了編譯你需要指定 /unsafe 編譯選項,如果你是使用的是Microsoft Visual Studio你需要在項目選項中把 'Allow unsafe code block'設置成 True。

C#操作內存之指針的基本用法

還有一些與指針緊密聯系的操作符,那就是 & 操作符,& 返回它所操作對象的地址。

例如:

unsafe   {  int* pi;  int x = 1;  pi = &x;  System.Console.WriteLine("Value of x is: " + *pi);  }

在這個例子中我們創建了2個變量,’pi’是指向int的指針,’x’是int,然后我們將’x’在內存中的地址賦予’pi’,理解我們放在 ’pi’ 變量中的是 ’x’的地址而不是’x’的值非常重要 (使用: pi = x 將返回錯誤 "Cannot implicitly convert type 'int' to 'int*'")

編譯后執行將會輸出:

Value of x is: 1

指針可以接受 null 值,也可能使用 void 指針類型,下面的代碼可以正常編譯:

unsafe   {  nt x = 10;  void* px = &x;  double *pd = (double*)px;  }

fixed 關鍵字和垃圾回收

在 C# 中使用指針需要比在 C++種更加注意。這是因為垃圾回收器(g.c.)會運行內存清理,在清理的過程中,g.c.會改變對象的物理內存位置,如果 g.c.改變了對象的位置指針將指向錯誤的內存位置。為了避免這樣的問題(已經與垃圾回收器連接),C# 包含 'fixed' 關鍵字. 它通知系統不要讓垃圾回收器重新部署對象。

'fixed' 示例:

// pt is a managed variable, subject to g.c.  Colour cl = new Colour();   // must use fixed to get address of cl.R  fixed ( int* pi = &cl.R)  {   *pi = 1;   }

初始化同一類型的多個指針:

fixed (byte* pb = sarr, pd = darr) {...}

C#操作內存之初始化不同類型的指針:

fixed (int* pi = &cl.G)  fixed (double* pd = &array[10])

如果我們忘了 ’fixed’ 關鍵字編譯器會給我們相應的警告,但它沒有智能到在下面的情況中也會警告我們。下面的代碼有一個嚴重的Bug盡管編譯很正常。

class Test  {  public int x;  }  unsafe class SimpleTest  {  [STAThread]  static void Main(string[] args)  {  Test test = new Test();  int* pi;  fixed (int* px = &test.x)  {  *px = 100;  pi = px;  }  Console.WriteLine("before g.c.: " + *pi);  System.GC.Collect(2);  Console.WriteLine("after g.c.: " + *pi);  }  }

在我的機器上結果是:

before g.c.: 100  after g.c.: 132

我們可以看到同一個指針有兩個不同的值,事實上在'before g.c.' 和 'after g.c.' 能得到不同結果的可能性非常小,because probability of starting garbage collector is very little. 但是作為一個規則我們應該避免在fixed塊以外使用指針,我們的情況是每次在fixed塊外使用 ’pi’ 指針都有可能產生難以診斷的錯誤。

C#操作內存之指針和WinApi

使用指針最重要的好處就是可以與其他二進制代碼進行交互。許多 WinApi 函數都使用指針,例如GetComputerName (Kernel32.lib.)可以提供我們的計算機的名稱。

BOOL GetComputerName(LPTSTR lpBuffer,   // computer name  LPDWORD lpnSize // size of name buffer);

下面的程序演示如何使用GetComputerName:

[System.Runtime.InteropServices.DllImport("Kernel32")]  static extern unsafe bool   GetComputerName(byte* lpBuffer,long* nSize);  static void Main()  {  byte[] buffor = new byte[512];  long size = buffor.Length;  unsafe {  long* pSize = &size;  fixed (byte* pBuffor = buffor)  {  GetComputerName(pBuffor,pSize);  }  }  System.Text.Encoding textEnc =   new System.Text.ASCIIEncoding();  System.Console.WriteLine(  "Computer name: {0}",textEnc.GetString(buffor));   }

C#操作內存結論

我們已經看到指針是C#語言中非常有用的部分,使用指針并不難但是要非常小心,因為有可能會導致難以診斷的問題,使用指針會擾亂垃圾回收器的功能,特別當我們在程序中大量使用指針。因此在之用指針之前我們應該多考慮,或者嘗試其他的解決辦法。

“C#操作內存的指針怎么理解”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注億速云網站,小編將為大家輸出更多高質量的實用文章!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

丰县| 名山县| 林西县| 旺苍县| 龙胜| 镇康县| 唐海县| 保亭| 泸溪县| 咸宁市| 白水县| 临汾市| 宁南县| 博罗县| 惠来县| 若尔盖县| 图们市| 永年县| 交口县| 客服| 三河市| 阿尔山市| 德安县| 仙居县| 满洲里市| 太保市| 申扎县| 炉霍县| 阜阳市| 东光县| 十堰市| 临西县| 商河县| 江城| 如东县| 托克逊县| 义马市| 靖远县| 闵行区| 民县| 横山县|