成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

.NET內存映射文件原理、創建及進程通訊

開發 后端
本文介紹了.NET內存映射文件的原理、創建以及實現進程通訊的方法。NET 4.0新增加了一個System.IO. MemoryMappedFiles命名空間,其中添加了幾個類和相應的枚舉類型,從而使我們可以很方便地創建內存映射文件。

操作系統很早就開始使用內存映射文件(Memory Mapped File)來作為進程間的共享存儲區,這是一種非常高效的進程通訊手段。Win32 API中也包含有創建內存映射文件的函數,然而,這些函數都運行于非托管環境下,在.NET中只能通過平臺調用機制來使用它們,用起來很不方便。幸運的是,.NET 4.0新增加了一個System.IO. MemoryMappedFiles命名空間,其中添加了幾個類和相應的枚舉類型,從而使我們可以很方便地創建內存映射文件。

1 內存映射文件原理

所謂內存映射文件,其實就是在內存中開辟出一塊存放數據的專用區域,這區域往往與硬盤上特定的文件相對應。進程將這塊內存區域映射到自己的地址空間中,訪問它就象是訪問普通的內存一樣。

內存映射文件原理圖 

圖 1 .NET內存映射文件原理圖

在.NET中,使用MemoryMappedFile對象表示一個內存映射文件,通過它的CreateFromFile()方法根據磁盤現有文件創建內存映射文件,調用這一方法需要提供一個與磁盤現有文件相對應的FileStream對象。

以下示例代碼動態創建一個MyFile.dat文件,然后將其映射到系統內存中,設定容量為1M:

  1. FileStream fs = new FileStream("MyFile.dat", FileMode.Create,  
  2. FileAccess.ReadWrite);  
  3. MemoryMappedFile memoryFile = MemoryMappedFile.CreateFromFile(fs, "MyFile", 1024*1024); 

注意用于創建內存映射文件的文件流必須是可讀寫的。

擴充閱讀:

關于內存映射文件的容量

默認情況下,在調用MemoryMappedFile.CreateFromFile()方法時如果不指定文件容量,那么,創建的內存映射文件的容量等同于文件的大小。

在上面的示例代碼中,由于磁盤文件是臨時生成的,其長度為0,所以,必須在創建內存映射文件時同時指定其容量。

在設定內存映射文件的容量時,其值不能小于磁盤文件的現有長度,但可以比它大。但要注意這將導致一個戲劇化的結果:磁盤文件自動增長到聲明的容量大小!

可以多次調用MemoryMappedFile.CreateFromFile(),每次傳給它一個更大的容量數值以不斷擴充磁盤文件的大小。

當不再使用一個MemoryMappedFile對象時,注意應該及時地調用其Dispose()方法釋放它所占有的系統資源。因為MemoryMappedFile實際上對應著運行操作系統核心的核心對象,如果不及時關閉,會造成操作系統核心資源(比如句柄)的浪費,要等到MemoryMappedFile對象被CLR垃圾回收,或者整個進程中止時,這些資源才會被操作系統回收再利用。

另外,內存映射文件的容量其實是指***允許分配給內存映射文件的內存存儲區字節數,并不意味著系統會馬上分配指定容量的內存。進程中訪問這塊映射到磁盤文件中的存儲區時,操作系統如果發現其內容還未裝入內存,就會從磁盤文件中裝入相應內容到內存中。因此,不用擔心聲明一個大的內存映射文件容量會導致內存的浪費。

當MemoryMappedFile對象創建之后,我們并不能直接對其進行讀寫,必須通過一個MemoryMappedViewAccessor對象來訪問這個內存映射文件。

MemoryMappedFile. CreateViewAccessor()方法可以創建MemoryMappedViewAccessor對象,而此對象提供了一系列讀寫的方法,用于向內存映射文件中讀取和寫入數據。

以下示例代碼創建了一個內存映射文件訪問對象并使用它寫入數據:

  1. FileStream fs =…;  //創建FileStream對象  
  2. MemoryMappedFile memoryFile=…;  //創建內存映射文件  
  3. //創建內存映射文件訪問對象  
  4. MemoryMappedViewAccessor accessor=  
  5. memoryFile.CreateViewAccessor(0, 1024);  
  6. for (int i = 0; i < 1024; i+=2)  
  7. accessor.Write(i, ‘c’);  

上述代碼中要注意,在創建內存映射文件訪問對象需要指定它所能訪問的內存映射文件的內容范圍,這個“范圍”稱為“內存映射視圖(Memory Mapped View)”。可以將它與“放大鏡”類比,當使用一個放大鏡閱讀書籍時,一次只能放大指定部分的文字。類似地,我們只能在內存映射視圖所規定的范圍內存取內存映射文件。

在上述代碼中,我們看到內存映射視圖對象accessor只提取了內存映射文件開頭1024個字節的內容,然后,向其中寫入了512個“c”字符。

當調用內存映射視圖對象的Write()方法時,需要指明從哪個位置(即方法的***個參數)開始寫入數據,并且需要計算清楚要寫入的數據占幾個字節,這樣,當寫入下一個數據時,就知道應該從哪個位置開始。

注意,Write()方法中的位置是相對視圖對象而非內存映射文件本身,因此,此位置數值再加上視圖距內存映射文件開頭的位置數據才是寫入的數據在文件中的真實位置。

Write()方法有多個重載形式,可以向內存映射文件中寫入多種類型的數據,但要注意計算清楚其寫入的位置,避免造成數據覆蓋問題。

類似地,內存映射視圖對象提供了多個重載的Read()方法,可以從內存映射文件中讀取數據。

比較有趣的是,在同一個進程中可以針對同一個內存映射文件創建多個視圖對象,從而允許我們同時修改同一個文件的不同部分,在關閉視圖對象時由操作系統保證將所有修改都寫回到原始文件中。

下面我們來看一個示例。

2 在同一進程內同時讀寫同一內存映射文件

示例項目UseMMFInProcess運行時會在程序的當前目錄下創建一個“MyFile.dat”文件,然后,創建了兩個內存映射視圖對象,分別向文件的前半部分和后半部分寫入不同的數據,然后再從中讀出來(圖 2)。

示例項目UseMMFInProcess 

圖 2 .NET內存映射文件示例項目UseMMFInProcess

這個示例展示的技術很基礎,請讀者自行查看源碼。

3 使用內存映射文件在進程間傳送值類型數據

在前面的例子中,內存映射文件直接與某個特定的磁盤文件相對應,事實上,我們也可以不用創建磁盤文件而直接使用Windows的分頁文件。這種方式是實現進程間互傳數據的典型方式。

調用MemoryMappedFile.CreateNew()或MemoryMappedFile.CreateOrOpen()方法可以在系統內存(System Memory)中直接創建一個內存映射文件,這個內存映射文件所對應的“物理文件”是Windows的系統分頁文件。兩個方法都需要給映射文件指定一個唯一的名稱。不同之處在于CreateOrOpen ()方法在指定名稱的映射文件存在時就直接將其返回給進程,而CreateNew()方法始終是新創建一個內存映射文件。

擴充閱讀:

Windows的系統分頁文件和休眠文件

默認情況下,在安裝Windows的分區根目錄下,會找到兩個具有“隱藏”屬性的pagefile.sys和hiberfil.sys文件,前者(pagefile.sys)就是Windows的分頁文件,用于保存從物理內存中換出的內存頁,我們可以用它的一部分來創建內存映射文件。后者(hiberfil.sys)則是“系統休眠”文件,當Windows啟用了休眠功能時,就會在硬盤上找到這個文件,它的內容是系統休眠時物理內存中的數據,當計算機從休眠中“醒”過來時,通過從此文件中加載信息以恢復上次工作的狀態。

內存映射文件創建好以后,可以如同前面介紹的方法一樣創建視圖對象,然后使用Read和Write系列方法存取。

只要指定同一個名字,那么,多個進程就可以使用同一個內存映射文件交換數據。示例UseMMFBetweenProcess展示了在兩個進程間相互交換一個結構體變量的情況:

示例項目UseMMFBetweenProcess 

圖 3 .NET內存映射文件示例項目UseMMFBetweenProcess

兩個進程要交換的數據格式如下:

  1. public struct MyStructure  
  2. {  
  3.  public int IntValue  
  4.  {  get;  set;   }  
  5.  public float FloatValue  
  6.  {  get;  set;   }  
  7. }  

啟動UseMMFBetweenProcess程序的兩個實例,在其中一個窗體上輸入兩個數字之后,點擊“保存”按鈕,然后在另一個進程的窗體上點擊“提取”,可以看到另一個進程寫入的信息出現在本進程的文本框中。

示例程序采用MemoryMappedFile.CreateOrOpen()方法創建或打開一個內存映射文件,然后調用MemoryMappedViewAccessor類的泛型方法Write<T>()和Read<T>()向內存映射文件中寫入和讀取數據。

注意,泛型方法Write<T>()和Read<T>()中的泛型參數T必須是值類型(比如整型int和結構體struct),特別地,對于用戶自定義的結構體,要求其成員也必須是值類型。

例如,以下結構體將無法寫入到內存映射文件中,因為其成員Info是string類型的,這是一個引用類型。

  1. public struct ErrorStruct  
  2. {  
  3.  public string Info;  

之所以要求泛型參數不能是引用類型,其道理非常簡單,如果結構體中的某個成員是引用類型,那么在程序運行時,計算機無法知道應該向內存映射文件中寫入多少個字節,因為引用類型的變量所引用的對象位于托管堆中,其占用存儲空間的大小不經過計算是難以確定的,而完成這個計算工作將花費不少的系統資源(想想一個對象可能又會引用到另一個對象就明白了),這會嚴重影響內存映射文件讀寫操作效率。

兩個進程不能交換引用類型的數據,這個限制似乎還不小,但事實上,我們完成可以通過對象序列化技術來突破這個限制,在兩個進程間交換任意大小的對象(只要內存映射文件有足夠的容量)。請看下一小節的示例UseMMFBetweenProcess2。

4 利用序列化技術通過內存映射文件實現進程通訊

示例:UseMMFBetweenProcess2 

圖4  .NET內存映射文件示例:UseMMFBetweenProcess2

如圖 4所示,運行示例程序的多個實例,加載圖片并輸入圖片說明,點擊相應按鈕后,可以在多個進程間直接交換以下格式的信息:

  1. [Serializable]  
  2. class MyPic  
  3. {  
  4.  public Image pic;//圖片  
  5.  public string picInfo;  //圖片信息說明  

請注意這是一個引用類型的數據對象,并且它附加了可序列化“[Serializable]”的代碼屬性。

如果要向內存映射文件中序列化對象,必須將內存映射文件轉換為可順序讀取的流。幸運的是,MemoryMappedFile類的CreateViewStream()方法可以創建一個MemoryMappedViewStream對象,通過它即可序列化對象,其代碼框架如下:

  1. //創建或打開內存映射文件  
  2. MemoryMappedFile memoryFile = MemoryMappedFile.CreateOrOpen(...);  
  3. //創建內存映射流  
  4. MemoryMappedViewStream stream = memoryFile.CreateViewStream();  
  5. //創建要在進程間交換的信息對象  
  6. MyPic obj =...;  
  7. //向內存映射流中序列化對象  
  8. IFormatter formatter = new BinaryFormatter();  
  9. stream.Seek(0, SeekOrigin.Begin);  
  10. formatter.Serialize(stream, obj); 

本文來自bitfan(數字世界一凡人)的專欄:《.NET 4.0中使用內存映射文件實現進程通訊》。

【編輯推薦】

  1. .NET內存管理的***實踐
  2. .NET 4.0內存映射文件詳解
  3. .NET真的不用管內存嗎?從List﹤T﹥列表聊起
  4. 淺析C#編程中的內存管理
  5. .NET 4.0 Beta 1新增STM特性概覽
責任編輯:yangsai 來源: bitfan的專欄
相關推薦

2009-07-24 10:00:38

.NET 4.0內存映

2009-02-19 13:28:08

遠程通訊技術及原理Java

2023-03-01 10:37:51

2011-04-25 17:15:39

MongodbMMAP

2012-06-20 14:16:36

Java內存映射

2018-11-30 09:03:55

HTTP緩存Web

2014-06-13 11:22:18

Https

2011-06-13 09:15:18

AIXlinuxunix

2023-12-07 12:45:58

進程共享數據

2021-04-27 13:56:49

內存.映射地址

2010-03-03 09:16:17

2023-08-03 07:30:01

JavaNIO庫

2019-09-02 14:53:53

JVM內存布局GC

2012-06-29 13:54:11

Java內存原型

2025-05-12 09:12:59

2011-06-16 10:27:55

.NET內存泄漏

2009-07-17 13:55:02

WinCE文件目錄

2024-03-20 10:48:09

Java 8內存管理

2023-02-07 08:55:04

進程棧內存底層

2017-08-06 00:05:18

進程通信開發
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 亚洲精品久久区二区三区蜜桃臀 | 毛片黄片| 日韩欧美在线不卡 | 欧美激情一区 | 久久国产传媒 | 亚洲乱码国产乱码精品精的特点 | 国产精品久久性 | 国产精品久久久久久久久久 | 成年人网站在线观看视频 | 亚洲人成人一区二区在线观看 | 欧美日韩综合一区 | 亚洲视频一区在线播放 | 97久久久久久久久 | 国产乱码精品一品二品 | 在线免费91 | 欧美中文字幕 | 手机看黄av免费网址 | 久久久久国产成人精品亚洲午夜 | www.黄色在线观看 | 97av在线| 玖玖综合在线 | 欧美日韩在线免费 | 天堂成人国产精品一区 | 日日摸日日碰夜夜爽2015电影 | 国产精品视频久久久 | 免费网站国产 | 国产高清一区二区 | 在线观看电影av | 欧美一区在线视频 | 黄色大片免费观看 | 欧美精品一二三区 | 亚洲精品自在在线观看 | 久久久久亚洲av毛片大全 | 国产精品99久久久久久www | 91精品国产91久久综合桃花 | 一区二区三区亚洲精品国 | 久久综合久色欧美综合狠狠 | 欧美色视频免费 | 天堂av免费观看 | 精品国产成人 | 中文字字幕在线中文乱码范文 |