首页 > 系统相关 >C#-exe间通讯(二)-共享内存

C#-exe间通讯(二)-共享内存

时间:2022-12-26 10:34:41浏览次数:40  
标签:exe 映射 C# shareInfo ShareInfo MemoryMappedFile 内存 共享内存 ViewAccessor

一、共享内存的帮助类

/**
*┌──────────────────────────────────────────────────────────────┐
*│ 描    述:两个exe程序之间通信_共享内存传递(基础示例,程序中使用时不进一步封装是不够的)                                                  
*│ 作    者:执笔小白                                              
*│ 版    本:1.0                                       
*│ 创建时间:2022-12-06 10:40:56                            
*└──────────────────────────────────────────────────────────────┘
*┌──────────────────────────────────────────────────────────────┐
*│ 命名空间: Communication_Between_exe_ProgramsHelper                              
*│ 类    名:Communication_SharedMemoryHelper                                     
*└──────────────────────────────────────────────────────────────┘
*/
using Newtonsoft.Json;
using System.IO.MemoryMappedFiles;
using System.Runtime.InteropServices;

namespace Communication_Between_exe_ProgramsHelper
{
    /// <summary>
    /// 两个exe程序之间通信_共享内存传递(主从运行,不可单独运行)
    /// </summary>
    public class Communication_SharedMemoryHelper
    {
        /*
         * 二进制 100=0x1100100
         * 
         * 注意一:线程或进程间通讯时使用互斥锁
         * //bool mutexCreated;
         * //Mutex mutex = newMutex(true, "testmapmutex", out mutexCreated);  // 创建锁(指定锁名)
         * //try{
         * //    mutex.WaitOne();  // 等待其他程序释放锁
         * //    // 通讯处理...
         * //}
         * //finally{
         * //    mutex.ReleaseMutex();  // 释放锁
         * //}
         * 
         * 注意二:非持久文件映射时,注意MemoryMappedFile不要被GC。
         * 
         */

        #region 非持久文件映射
        /*
         * 1、非持久文件映射
         * 非持久文件是未与磁盘上的源文件关联的内存映射文件。
         * 当最后一个进程使用完此文件后,数据将丢失,并且垃圾回收功能将回收此文件。
         * 可使用非托管对象防止被回收
         */

        private static MemoryMappedFile? mmf_NoKeep;

        /// <summary>
        /// 内存共享_基于流的操作_读-非持久文件映射
        /// </summary>
        /// <param name="mmfName">共享内容对象名</param>
        /// <returns></returns>
        public static MemoryMappedFile_ShareInfo_ViewStream MemoryMappedFile_ViewStream_Read_NoKeep(string mmfName)
        {
            MemoryMappedFile_ShareInfo_ViewStream resultShareInfo = new();

            mmf_NoKeep = MemoryMappedFile.CreateOrOpen(mmfName, 0x1100100, MemoryMappedFileAccess.ReadWrite);  // 打开共享内存区域,大小默认为100MB
            using (MemoryMappedViewStream stream = mmf_NoKeep.CreateViewStream())
            {
                if (stream.CanRead)
                {
                    BinaryReader reader = new(stream);
                    if (JsonConvert.DeserializeObject<MemoryMappedFile_ShareInfo_ViewStream>(reader.ReadString()) is MemoryMappedFile_ShareInfo_ViewStream share)
                    {
                        resultShareInfo = share;
                    }
                }
            }
            return resultShareInfo;
        }

        /// <summary>
        /// 内存共享_基于内存偏移量的操作_读-非持久文件映射
        /// </summary>
        /// <param name="mmfName">共享内容对象名</param>
        /// <returns></returns>
        public static MemoryMappedFile_ShareInfo_ViewAccessor MemoryMappedFile_ViewAccessor_Read_NoKeep(string mmfName)
        {
            MemoryMappedFile_ShareInfo_ViewAccessor resultShareInfo = new();

            mmf_NoKeep = MemoryMappedFile.CreateOrOpen(mmfName, 0x1100100, MemoryMappedFileAccess.ReadWrite);  // 打开共享内存区域,大小默认为100MB
            using (MemoryMappedViewAccessor accessor = mmf_NoKeep.CreateViewAccessor(0, 100))  // 注意这里的偏移量,大小;大小100MB
            {
                int colorSize = Marshal.SizeOf(typeof(MemoryMappedFile_ShareInfo_ViewAccessor));
                accessor.Read(0, out resultShareInfo);  // 读取视图
            }
            return resultShareInfo;
        }

        /// <summary>
        /// 内存共享_基于流的操作_写-非持久文件映射
        /// </summary>
        /// <param name="mmfName">共享内容对象名</param>
        /// <param name="share">写入的内容</param>
        public static bool MemoryMappedFile_ViewStream_Wirte_NoKeep(string mmfName, MemoryMappedFile_ShareInfo_ViewStream shareInfo)
        {
            bool result = false;

            mmf_NoKeep = MemoryMappedFile.CreateOrOpen(mmfName, 0x1100100, MemoryMappedFileAccess.ReadWrite);  // 打开共享内存区域,大小默认为100MB
            using (MemoryMappedViewStream stream = mmf_NoKeep.CreateViewStream())
            {
                result = stream.CanWrite;
                if (result)
                {
                    BinaryWriter writer = new(stream);
                    writer.Write(JsonConvert.SerializeObject(shareInfo));
                }
            }
            return result;
        }

        /// <summary>
        /// 内存共享_基于内存偏移量的操作_写-非持久文件映射
        /// </summary>
        /// <param name="mmfName"></param>
        /// <param name="shareInfo"></param>
        /// <returns></returns>
        public static bool MemoryMappedFile_ViewAccessor_Wirte_NoKeep(string mmfName, MemoryMappedFile_ShareInfo_ViewAccessor shareInfo)
        {
            bool result = false;

            mmf_NoKeep = MemoryMappedFile.CreateOrOpen(mmfName, 0x1100100, MemoryMappedFileAccess.ReadWrite);  // 打开共享内存区域,大小默认为100MB 
            using (MemoryMappedViewAccessor accessor = mmf_NoKeep.CreateViewAccessor(0, 100))  // 注意这里的偏移量,大小;大小100MB
            {
                result = accessor.CanRead;
                if (result)
                {
                    int colorSize = Marshal.SizeOf(typeof(MemoryMappedFile_ShareInfo_ViewAccessor));
                    // 视图操作
                    accessor.Write(0, ref shareInfo);  // 修改视图
                }
            }
            return result;
        }
        #endregion 非持久文件映射

        #region 持久文件映射
        /* 
         * 2、持久文件映射
         * 持久文件是与磁盘上的源文件关联的内存映射文件。
         * 在最后一个进程使用完此文件后,数据将保存到磁盘上的源文件中。这些内存映射文件适合用来处理非常大的源文件。
         * 不需要考虑维护共享内存对象被回收的问题
         */

        /// <summary>
        /// 内存共享_基于流的操作_读-持久文件映射
        /// </summary>
        /// <param name="mmfFile">共享内容对象的文件路径</param>
        /// <param name="mmfName">共享内容对象名</param>
        /// <returns></returns>
        public static MemoryMappedFile_ShareInfo_ViewStream MemoryMappedFile_ViewStream_Read_Keep(string mmfFile, string mmfName, long defaultSize = 0x1100100)
        {
            MemoryMappedFile_ShareInfo_ViewStream resultShareInfo = new();

            using (MemoryMappedFile mmf_Keep = MemoryMappedFile.CreateFromFile(mmfFile, FileMode.OpenOrCreate, mmfName, defaultSize))  // "D:\\内存映射文件.data",默认100MB
            {
                using (MemoryMappedViewStream stream = mmf_Keep.CreateViewStream())
                {
                    if (stream.CanRead)
                    {
                        BinaryReader reader = new(stream);
                        if (JsonConvert.DeserializeObject<MemoryMappedFile_ShareInfo_ViewStream>(reader.ReadString()) is MemoryMappedFile_ShareInfo_ViewStream share)
                        {
                            resultShareInfo = share;
                        }
                    }
                }
            }
            return resultShareInfo;
        }

        /// <summary>
        /// 内存共享_基于内存偏移量的操作_读-持久文件映射
        /// </summary>
        /// <param name="mmfFile">共享内容对象的文件路径</param>
        /// <param name="mmfName">共享内容对象名</param>
        /// <returns></returns>
        public static MemoryMappedFile_ShareInfo_ViewAccessor MemoryMappedFile_ViewAccessor_Read_Keep(string mmfFile, string mmfName, long defaultSize = 0x1100100)
        {
            MemoryMappedFile_ShareInfo_ViewAccessor resultShareInfo = new();

            using (MemoryMappedFile mmf_Keep = MemoryMappedFile.CreateFromFile(mmfFile, FileMode.OpenOrCreate, mmfName, defaultSize))  // "D:\\内存映射文件.data",默认100MB
            {
                using (MemoryMappedViewAccessor accessor = mmf_Keep.CreateViewAccessor(0, 100))  // 注意这里的偏移量,大小;大小需要调整
                {
                    int colorSize = Marshal.SizeOf(typeof(MemoryMappedFile_ShareInfo_ViewAccessor));
                    // 视图操作
                    accessor.Read(0, out resultShareInfo);  // 读取视图
                }
            }

            return resultShareInfo;
        }

        /// <summary>
        /// 内存共享_基于流的操作_写-持久文件映射
        /// </summary>
        /// <param name="mmfFile">共享内容对象的文件路径</param>
        /// <param name="mmfName">共享内容对象名</param>
        /// <param name="share">写入的内容</param>
        public static bool MemoryMappedFile_ViewStream_Wirte_Keep(string mmfFile, string mmfName, MemoryMappedFile_ShareInfo_ViewStream shareInfo, long defaultSize = 0x1100100)
        {
            bool result = false;
            using (MemoryMappedFile mmf_Keep = MemoryMappedFile.CreateFromFile(mmfFile, FileMode.OpenOrCreate, mmfName, defaultSize))  // "D:\\内存映射文件.data",默认100MB
            {
                using (MemoryMappedViewStream stream = mmf_Keep.CreateViewStream())
                {
                    result = stream.CanWrite;
                    if (result)
                    {
                        BinaryWriter writer = new(stream);
                        writer.Write(JsonConvert.SerializeObject(shareInfo));
                    }
                }
            }
            return result;
        }

        /// <summary>
        /// 内存共享_基于内存偏移量的操作_写-持久文件映射
        /// </summary>
        /// <param name="mmfFile">共享内容对象的文件路径</param>
        /// <param name="mmfName">共享内容对象名</param>
        /// <param name="share">写入的内容(二进制,单位MB)</param>
        public static bool MemoryMappedFile_ViewAccessor_Wirte_Keep(string mmfFile, string mmfName, MemoryMappedFile_ShareInfo_ViewAccessor shareInfo, long defaultSize = 0x1100100)
        {
            bool result = false;

            using (MemoryMappedFile mmf_Keep = MemoryMappedFile.CreateFromFile(mmfFile, FileMode.OpenOrCreate, mmfName, defaultSize))  // "D:\\内存映射文件.data",默认100MB
            {
                using (MemoryMappedViewAccessor accessor = mmf_Keep.CreateViewAccessor(0, 100))  // 注意这里的偏移量,大小;大小需要调整
                {
                    result = accessor.CanRead;
                    if (result)
                    {
                        int colorSize = Marshal.SizeOf(typeof(MemoryMappedFile_ShareInfo_ViewAccessor));
                        // 视图操作
                        accessor.Write(0, ref shareInfo);  // 修改视图
                    }
                }
            }
            return result;
        }
        #endregion 持久文件映射

        /// <summary>
        /// 共享内存_基于流的
        /// </summary>
        public class MemoryMappedFile_ShareInfo_ViewStream
        {
            /// <summary>
            /// 标识ID
            /// </summary>
            public string SessionIdGUID { set; get; } = string.Empty;

            /// <summary>
            /// 创建时间
            /// </summary>
            public DateTime CreateTime { set; get; } = DateTime.Now;

            /// <summary>
            /// 自定义的共享信息
            /// </summary>
            public object InfoData { set; get; } = new object();
        }

        /// <summary>
        /// 共享内存_基于偏移量的
        /// </summary>
        public struct MemoryMappedFile_ShareInfo_ViewAccessor
        {
            /// <summary>
            /// 标识ID
            /// </summary>
            public int SessionIdGUID { set; get; } = 0;

            /// <summary>
            /// 创建时间
            /// </summary>
            public DateTime CreateTime { set; get; } = DateTime.Now;

            /// <summary>
            /// 自定义的共享信息
            /// </summary>
            public long InfoData { set; get; } = 0;

            public MemoryMappedFile_ShareInfo_ViewAccessor()
            {

            }
        }
    }
}

二、发送端:

        /// <summary>
        /// 共享内存传递_发送
        /// </summary>
        private void btnSedMsg_SharedMemo_Click(object sender, EventArgs e)
        {
            bool result = false;

            if (radIsStream.Checked && chkIsKeep.Checked)
            {
                MemoryMappedFile_ShareInfo_ViewStream shareInfo = new();
                shareInfo.SessionIdGUID = "1";
                shareInfo.CreateTime = DateTime.Now;
                shareInfo.InfoData = txtMsg_SharedMemo.Text.Trim();

               // result = MemoryMappedFile_ViewStream_Wirte_Keep(txtShareFilePath.Text.Trim(), txtShareName.Text.Trim(), shareInfo); 
               result = MemoryMappedFile_ViewStream_Wirte_Keep(@"D:\ShareTest.data", @"ImgA", shareInfo);
            }  // 内存共享_基于流的操作_写-持久文件映射
            else if (radIsStream.Checked && !chkIsKeep.Checked)
            {
                MemoryMappedFile_ShareInfo_ViewStream shareInfo = new();
                shareInfo.SessionIdGUID = "1";
                shareInfo.CreateTime = DateTime.Now;
                shareInfo.InfoData = txtMsg_SharedMemo.Text.Trim();

                result = MemoryMappedFile_ViewStream_Wirte_NoKeep(txtShareName.Text.Trim(), shareInfo);
            }  // 内存共享_基于流的操作_写-非持久文件映射
            else if (!radIsStream.Checked && chkIsKeep.Checked)
            {
                MemoryMappedFile_ShareInfo_ViewAccessor shareInfo = new();
                shareInfo.SessionIdGUID = 1;
                shareInfo.CreateTime = DateTime.Now;
                shareInfo.InfoData = Convert.ToInt64(txtMsg_SharedMemo.Text.Trim());

                //result = MemoryMappedFile_ViewAccessor_Wirte_Keep(txtShareFilePath.Text.Trim(), txtShareName.Text.Trim(), shareInfo);
                result = MemoryMappedFile_ViewAccessor_Wirte_Keep(@"D:\ShareTest.data", @"ImgA", shareInfo);
            }  // 内存共享_基于内存偏移量的操作_写-持久文件映射
            else if (!radIsStream.Checked && !chkIsKeep.Checked)
            {
                MemoryMappedFile_ShareInfo_ViewAccessor shareInfo = new();
                shareInfo.SessionIdGUID = 1;
                shareInfo.CreateTime = DateTime.Now;
                shareInfo.InfoData = Convert.ToInt64(txtMsg_SharedMemo.Text.Trim());

                result = MemoryMappedFile_ViewAccessor_Wirte_NoKeep(txtShareName.Text.Trim(), shareInfo);  //
            }  // 内存共享_基于内存偏移量的操作_写-非持久文件映射
            txtMsg_SharedMemo.Text += ";" + result.ToString();
        }

三、接收端:

        /// <summary>
        /// 共享内存方式-手动获取(根据需要加互斥锁或进程内线程锁)
        /// </summary>
        private void BtnManualGet_Click(object sender, EventArgs e)
        {
            if (radIsStream.Checked && chkIsKeep.Checked)  // 内存共享_基于流的操作_写-持久文件映射
            {
                MemoryMappedFile_ShareInfo_ViewStream shareInfo = MemoryMappedFile_ViewStream_Read_Keep(txtShareFilePath.Text.Trim(), txtShareName.Text.Trim());
                if (shareInfo != null && shareInfo.InfoData is string infoData)
                {
                    rTxtShareMemo_msg.Text = "信息序号:" + shareInfo.SessionIdGUID + ";发送时间:" + shareInfo.CreateTime + ";内容:" + (!string.IsNullOrEmpty(infoData) ? infoData : string.Empty);
                }
                else
                {
                    rTxtShareMemo_msg.Text = "未获取到数据!";
                }
            }
            else if (radIsStream.Checked && !chkIsKeep.Checked)  // 内存共享_基于流的操作_写-非持久文件映射
            {
                MemoryMappedFile_ShareInfo_ViewStream shareInfo = MemoryMappedFile_ViewStream_Read_NoKeep(txtShareName.Text.Trim());
                if (shareInfo != null && shareInfo.InfoData is string infoData)
                {
                    rTxtShareMemo_msg.Text = "信息序号:" + shareInfo.SessionIdGUID + ";发送时间:" + shareInfo.CreateTime + ";内容:" + (!string.IsNullOrEmpty(infoData) ? infoData : string.Empty);
                }
                else
                {
                    rTxtShareMemo_msg.Text = "未获取到数据!";
                }
            }
            else if (!radIsStream.Checked && chkIsKeep.Checked)  // 内存共享_基于内存偏移量的操作_写-持久文件映射
            {
                MemoryMappedFile_ShareInfo_ViewAccessor shareInfo = MemoryMappedFile_ViewAccessor_Read_Keep(txtShareFilePath.Text.Trim(), txtShareName.Text.Trim());
                rTxtShareMemo_msg.Text = "信息序号:" + shareInfo.SessionIdGUID + ";发送时间:" + shareInfo.CreateTime + ";内容:" + (long)shareInfo.InfoData;
            }
            else if (!radIsStream.Checked && !chkIsKeep.Checked)  // 内存共享_基于内存偏移量的操作_写-非持久文件映射
            {
                MemoryMappedFile_ShareInfo_ViewAccessor shareInfo = MemoryMappedFile_ViewAccessor_Read_NoKeep(txtShareName.Text.Trim());  //

                rTxtShareMemo_msg.Text = "信息序号:" + shareInfo.SessionIdGUID + ";发送时间:" + shareInfo.CreateTime + ";内容:" + (long)shareInfo.InfoData;
            }
        }

标签:exe,映射,C#,shareInfo,ShareInfo,MemoryMappedFile,内存,共享内存,ViewAccessor
From: https://www.cnblogs.com/qq2806933146xiaobai/p/17005153.html

相关文章

  • Android真机调试手动添加程序包的LogCat
    android真机调试有时候看LogCat 时,有时候那个跑的本程序的LogCat 没有出现而是 出现的是"All messages(nofilters)"。此时的Log显示是显示的所有的信息,有时候需......
  • linux下close 掉socket 之后 阻塞的recv 不会立即返回
    在开发的一个基于rtmp聊天的程序时发现了一个很奇怪的现象。在windows下当我们执行closesocket 的操作之后,阻塞的recv 会立即返回-1。而在linux 下 当我们执行 ......
  • fatal error C1010: 在查找预编译头时遇到意外的文件结尾
    错误描述:fatalerrorC1010:在查找预编译头时遇到意外的文件结尾。是否忘记了向源中添加“#include"stdafx.h"”?错误分析:    此错误发生的原因是编译器在寻找预......
  • 插件丨PR/AE/FCPX视频磨皮神器Beauty Box超强一键磨皮
    DigitalAnarchy比较出众的一款视频磨皮美白降噪插件支持系统:windows、Mac软件版本:PR/AE:CS6—2023DavinciResolve达芬奇11以上FCPX10.1.2或者更高版本插件语言:WIN汉......
  • 既然Excel这么好用,为什么大公司都在用BI?一文带你搞懂奥秘!
    老李从事数据分析十几年了,也算是从Excel工具一路用到了BI工具,这里和大家分享一些Excel和BI之间的关系,正在犹豫是深入学习Excel还是开始学习BI的朋友们,相信这篇文章对你一定......
  • Webpack开发环境要如何搭建?
    本文首发自「慕课网」,想了解更多IT干货内容,程序员圈内热闻,欢迎关注!作者|慕课网精英讲师三水清本文将包括安装Node、NPM、WebPack开发环境,我使用的开发IDE是VScode。......
  • CodeQL的自动化代码审计之路(下篇)
    0x01前言在上一篇文章中,我们基于CodeQL官方提供的sdk实现了自动化查询数据库功能,在文章中也提到实现完整的自动化代码审计还缺少“数据库生成”相应的功能,本文主要针对“数......
  • SDL_AudioSpec
    这篇主要讲把视频的声音播放出来audioStream=-1;for(i=0;i<pFormatCtx->nb_streams;i++){if(pFormatCtx->streams[i]->codec->codec_type==......
  • rtmp发送H264及aac的音视频
    RTMP推送的音视频流的封装形式和FLV格式相似,由此可知,向FMS推送H264和AAC直播流,需要首先发送"AVCsequenceheader"和"AACsequenceheader",这两项数据包含的是重要的编码信......
  • 使用Dockerfile 制作docker 镜像
    下面是一个Dockerfile文件#基准镜像FROMcentos:7#作者信息MAINTAINER"aliyun"#工作目录WORKDIR/usr/local/src/#定义环境变量ENVNG_VERSIONnginx-1.21.0#......