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相同,但读取方法不同。
- App1代码:
- using System;
- using System.Collections.Generic;android从资源文件中读取文件流显示
- using System.Linq;
- using System.Text;
- using System.IO;
- //引用内存映射文件命名空间
- using System.IO.MemoryMappedFiles;
- namespace App1
- {
- class Program
- {
- static void Main(string[] args)
- {
- long capacity = 1<<10<<10;
- //创建或者打开共享内存
- using (var mmf = MemoryMappedFile.CreateOrOpen("testMmf", capacity, MemoryMappedFileAccess.ReadWrite))
- {
- //通过MemoryMappedFile的CreateViewAccssor方法获得共享内存的访问器
- var viewAccessor = mmf.CreateViewAccessor(0, capacity);
- //循环写入,使在这个进程中可以向共享内存中写入不同的字符串值
- while (true)
- {
- Console.WriteLine("请输入一行要写入共享内存的文字:");
- string input = Console.ReadLine();
- //向共享内存开始位置写入字符串的长度
- viewAccessor.Write(0, input.Length);
- //向共享内存4位置写入字符
- viewAccessor.WriteArray<char>(4, input.ToArray(), 0, input.Length);
- }
- }
- }
- }
- }
- App2代码:
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Text;
- using System.Threading;
- //引用使用内存映射文件需要的命名空间
- using System.IO.MemoryMappedFiles;
- namespace App2
- {
- class Program
- {
- static void Main(string[] args)
- {
- long capacity = 1<<10<<10;
- using (var mmf = MemoryMappedFile.OpenExisting("testMmf"))
- {
- MemoryMappedViewAccessor viewAccessor = mmf.CreateViewAccessor(0, capacity);
- //循环刷新共享内存字符串的值
- while (true)
- {
- //读取字符长度
- int strLength = viewAccessor.ReadInt32(0);
- char[] charsInMMf = new char[strLength];
- //读取字符
- viewAccessor.ReadArray<char>(4, charsInMMf, 0, strLength);
- Console.Clear();
- Console.Write(charsInMMf);
- Console.Write("\r");
- Thread.Sleep(200);
- }
- }
- }
- }
- }
- App3代码:
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Text;
- using System.IO.MemoryMappedFiles;
- using System.IO;
- namespace App3
- {
- class Program
- {
- static void Main(string[] args)
- {
- long capacity = 1 << 10 << 10;
- //打开共享内存
- using (var mmf = MemoryMappedFile.OpenExisting("testMmf"))
- {
- //使用CreateViewStream方法返回stream实例
- using (var mmViewStream = mmf.CreateViewStream(0, capacity))
- {
- //这里要制定Unicode编码否则会出问题
- using (BinaryReader rdr = new BinaryReader(mmViewStream,Encoding.Unicode))
- {
- while (true)
- {
- mmViewStream.Seek(0, SeekOrigin.Begin);
- int length = rdr.ReadInt32();
- char[] chars = rdr.ReadChars(length);
- Console.Write(chars);
- Console.Write("\r");
- System.Threading.Thread.Sleep(200);
- Console.Clear();
- }
- }
- }
- }
- }
- }
- }
在读数据时用了2种方法。
因为在之前很少会用到进程之间的通信,所以此方法只是想初步的认识下。此程序写的过于简陋,有很多东西都没有去判断。比如说是怎么创建了一个共享内存怎么取删除它等等。。。
希望我与此篇博文的作者共勉吧。
c#实现内存映射文件共享内存的更多相关文章
- System.IO之内存映射文件共享内存
内存映射文件是利用虚拟内存把文件映射到进程的地址空间中去,在此之后进程操作文件,就 像操作进程空间里的地址一样了,比如使用c语言的memcpy等内存操作的函数.这种方法能够很好的应用在需要频繁处理一个 ...
- kmalloc分配物理内存与高端内存映射--Linux内存管理(十八)
1 前景回顾 1.1 内核映射区 尽管vmalloc函数族可用于从高端内存域向内核映射页帧(这些在内核空间中通常是无法直接看到的), 但这并不是这些函数的实际用途. 重要的是强调以下事实 : 内核提供 ...
- linux编程之内存映射
一.概述 内存映射是在调用进程的虚拟地址空间创建一个新的内存映射. 内存映射分为2种: 1.文件映射 ...
- 使用Java内存映射(Memory-Mapped Files)处理大文件
>>NIO中的内存映射 (1)什么是内存映射文件内存映射文件,是由一个文件到一块内存的映射,可以理解为将一个文件映射到进程地址,然后可以通过操作内存来访问文件数据.说白了就是使用虚拟内存将 ...
- 《Java核心技术卷二》笔记(二)文件操作和内存映射文件
文件操作 上一篇已经总结了流操作,其中也包括文件的读写.文件系统除了读写以为还有很多其他的操作,如复制.移动.删除.目录浏览.属性读写等.在Java7之前,一直使用File类用于文件的操作.Java7 ...
- Java NIO内存映射---上G大文件处理(转)
林炳文Evankaka原创作品.转载请注明出处http://blog.csdn.net/evankaka 摘要:本文主要讲了java中内存映射的原理及过程,与传统IO进行了对比,最后,用实例说明了结果 ...
- Win 内存映射和堆栈
内存映射和堆栈 内存映射文件 内存映射文件可以用于3个不同的目的: 系统使用内存映射文件,以便加载和执行.exe和DLL文件.这可以大大节省页文件空间和应用程序启动运行所需的时间. 可以使用内存映射文 ...
- ZT 匿名内存映射
mmap函数使用 分类: Linux/Unix C/C++ 2008-01-22 17:03 6089人阅读 评论(1) 收藏 举报 unix编程null网络 UNIX网络编程第二卷进程间通信对mma ...
- 【VS开发】内存映射文件3
内存映射文件 内存映射文件,是由一个文件到一块内存的映射.Win32提供了允许应用程序把文件映射到一个进程的函数 (CreateFileMapping).内存映射文件与虚拟内存有些类似,通过内存映射文 ...
随机推荐
- POJ 2370
//我的解题思路是先把输入的含有n个元素的数组a排序(从小到大),然后对前(n+1)/2个元素作如下的处理, //s+= (a[i]+1)/2 #include <iostream> #i ...
- Docker学习--Linux基础准备篇
1.docker命令不需要附带敲sudo的解决办法 由于docker daemon需要绑定到主机的Unix socket而不是普通的TCP端口,而Unix socket的属主为root用户,所以其他用 ...
- Linux Cluster环境下批量分发执行补丁
转自:http://blog.csdn.net/napolunyishi/article/details/18219867 这两天做了一个需求,因为上一个版本的/tmp空间默认只分配了5G,而升级程序 ...
- Python -- 网络编程 -- 抓取网页图片 -- 图虫网
字符串(str)编码成字节码(bytes),字节码解码为字符串 获取当前环境编码:sys.stdin.encoding url编码urllib.parse.quote() url解码urllib.pa ...
- Kubernetes中的垃圾回收机制
本文所讨论垃圾回收(GC,Garbage Collection)机制针对Kubernetes1.1.7,docker容器. 一.Tips 01. Kubernetes的垃圾回收由kubelet进行管理 ...
- Eclipse中Maven项目出现红色感叹号问题
在Eclipse环境下,有时我们会遇到Maven项目出现红色感叹号的情形,而一旦项目出现感叹号,Eclipse便无法帮我们进行自动编译等工作,尽管有时候不会影响运行,但每次只能手动启动Maven重新编 ...
- C#合并两个Dictionary的方法
直接代码: public Dictionary<string, string> MergeDictionary(Dictionary<string, string> first ...
- 虚拟化明星——深挖轻量级容器docker
docker是一个轻量级容器,属于操作系统层面的虚拟化技术,封装了文件系统(AUFS)以及网络互联,进程隔离等特性. 传统虚拟化架构: docker虚拟化架构: 可以看出,docker是没有Guest ...
- activemq的三种通信方式
一.安装与启动 1.下载安装activemq,下载地址:http://activemq.apache.org/download.html. 2.安装完成后,进入其所在目录的bin目录下面,根据系统位数 ...
- js 3行代码,最简易实现div效果悬浮
简易实现浮动效果的首要因素是:获取滚动条距离浏览器顶部的距离,下面直接贴代码: <!DOCTYPE html> <html> <head> <meta cha ...