c#实现内存映射文件共享内存

原文:http://blog.csdn.net/wangtiewei/article/details/51112668

内存映射文件是利用虚拟内存把文件映射到进程的地址空间中去,在此之后进程操作文件,就像操作进程空间里的地址一样了,比如使用C语言的 memcpy等内存操作的函数。这种方法能够很好的应用在需要频繁处理一个文件或者是一个大文件的场合,这种方式处理IO效率比普通IO效率要高

共享内存是内存映射文件的一种特殊情况,内存映射的是一块内存,而非磁盘上的文件。共享内存的主语是进程(Process),操作系统默认会给每一 个进程分配一个内存空间,每一个进程只允许访问操作系统分配给它的哪一段内存,而不能访问其他进程的。而有时候需要在不同进程之间访问同一段内存,怎么办 呢?操作系统给出了创建访问共享内存的API,需要共享内存的进程可以通过这一组定义好的API来访问多个进程之间共有的内存,各个进程访问这一段内存就 像访问一个硬盘上的文件一样。而.Net 4.0中引入了System.IO. MemoryMappedFiles命名空间,这个命名空间的类对windows 共享内存相关API做了封装,使.Net程序员可以更方便的使用内存映射文件。

在C#中使用共享内存。以下App1的代码让用户输入一行文本到共享内存中;App2不停的刷新控制台,输出最新的共享内存内容;App3实现的功能和App2相同,但读取方法不同。

[csharp]view plaincopy

  1. App1代码:
  2. using System;
  3. using System.Collections.Generic;android从资源文件中读取文件流显示
  4. using System.Linq;
  5. using System.Text;
  6. using System.IO;
  7. //引用内存映射文件命名空间
  8. using System.IO.MemoryMappedFiles;
  9. namespace App1
  10. {
  11. class Program
  12. {
  13. static void Main(string[] args)
  14. {
  15. long capacity = 1<<10<<10;
  16. //创建或者打开共享内存
  17. using (var mmf = MemoryMappedFile.CreateOrOpen("testMmf", capacity, MemoryMappedFileAccess.ReadWrite))
  18. {
  19. //通过MemoryMappedFile的CreateViewAccssor方法获得共享内存的访问器
  20. var viewAccessor = mmf.CreateViewAccessor(0, capacity);
  21. //循环写入,使在这个进程中可以向共享内存中写入不同的字符串值
  22. while (true)
  23. {
  24. Console.WriteLine("请输入一行要写入共享内存的文字:");
  25. string input = Console.ReadLine();
  26. //向共享内存开始位置写入字符串的长度
  27. viewAccessor.Write(0, input.Length);
  28. //向共享内存4位置写入字符
  29. viewAccessor.WriteArray<char>(4, input.ToArray(), 0, input.Length);
  30. }
  31. }
  32. }
  33. }
  34. }

[csharp]view plaincopy

  1. App2代码:
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading;
  7. //引用使用内存映射文件需要的命名空间
  8. using System.IO.MemoryMappedFiles;
  9. namespace App2
  10. {
  11. class Program
  12. {
  13. static void Main(string[] args)
  14. {
  15. long capacity = 1<<10<<10;
  16. using (var mmf = MemoryMappedFile.OpenExisting("testMmf"))
  17. {
  18. MemoryMappedViewAccessor viewAccessor = mmf.CreateViewAccessor(0, capacity);
  19. //循环刷新共享内存字符串的值
  20. while (true)
  21. {
  22. //读取字符长度
  23. int strLength = viewAccessor.ReadInt32(0);
  24. char[] charsInMMf = new char[strLength];
  25. //读取字符
  26. viewAccessor.ReadArray<char>(4, charsInMMf, 0, strLength);
  27. Console.Clear();
  28. Console.Write(charsInMMf);
  29. Console.Write("\r");
  30. Thread.Sleep(200);
  31. }
  32. }
  33. }
  34. }
  35. }

[csharp]view plaincopy

  1. App3代码:
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.IO.MemoryMappedFiles;
  7. using System.IO;
  8. namespace App3
  9. {
  10. class Program
  11. {
  12. static void Main(string[] args)
  13. {
  14. long capacity = 1 << 10 << 10;
  15. //打开共享内存
  16. using (var mmf = MemoryMappedFile.OpenExisting("testMmf"))
  17. {
  18. //使用CreateViewStream方法返回stream实例
  19. using (var mmViewStream = mmf.CreateViewStream(0, capacity))
  20. {
  21. //这里要制定Unicode编码否则会出问题
  22. using (BinaryReader rdr = new BinaryReader(mmViewStream,Encoding.Unicode))
  23. {
  24. while (true)
  25. {
  26. mmViewStream.Seek(0, SeekOrigin.Begin);
  27. int length = rdr.ReadInt32();
  28. char[] chars = rdr.ReadChars(length);
  29. Console.Write(chars);
  30. Console.Write("\r");
  31. System.Threading.Thread.Sleep(200);
  32. Console.Clear();
  33. }
  34. }
  35. }
  36. }
  37. }
  38. }
  39. }

在读数据时用了2种方法。

因为在之前很少会用到进程之间的通信,所以此方法只是想初步的认识下。此程序写的过于简陋,有很多东西都没有去判断。比如说是怎么创建了一个共享内存怎么取删除它等等。。。

希望我与此篇博文的作者共勉吧。