2019-03-21 【c++&c#】进程间通讯__共享内存

进程间通讯方式:共享内存,管道(linux),udp通讯(若是在同一台电脑上通过udp通讯,那么它并没有经过网络,而是利用了网卡上的一块内存,而且非常可靠。)

这是项目中用到的一处共享内存,别人写的。记录一下,让每个看似毫无意义的项目都能吸收到一点有用的东西。
项目中使用到的地方:服务器要发送数据给unity,但是接收数据的格式是需要以固定的MFC(c/c++)代码接收。然后就在MFC代码中增加一处共享内存,作为中转站,unity读取这处共享内存获得数据。

  • 0.前置准备

| 句柄数据结构

        typedef struct semps
        {
            HANDLE hMFCWrite;                            //声明信号量变量
            HANDLE hU3DRead;                             //声明信号量变量
            HANDLE hU3DWrite;                            //声明信号量变量
            HANDLE hMFCRead;                             //声明信号量变量

            char *pBuf; 
            CMainFrame *mFrm;
        }Semps;

        Semps mSemp;
        HANDLE hMapFile;
  • 1.实例化共享内存

image.png
ProcessMsg::ProcessMsg(CMainFrame *frm)
{

    mSemp.hMFCWrite = CreateSemaphoreA(NULL,1,1,"WriteMap_1");
    mSemp.hU3DRead = CreateSemaphoreA(NULL,0,1,"ReadMap_1"); //第二各参数为0,代表进程创建信号量之时便声明了对该信号量的占用

    mSemp.hU3DWrite = CreateSemaphoreA(NULL,1,1,"WriteMap_2");
    mSemp.hMFCRead = CreateSemaphoreA(NULL,0,1,"ReadMap_2"); //第二各参数为0,代表进程创建信号量之时便声明了对该信号量的占用


    //创建共享内存区
    char szName[] = "shareMemory";    // 共享内存的名字

    // 创建共享文件句柄
    hMapFile = CreateFileMapping(
        INVALID_HANDLE_VALUE,    // 物理文件句柄
        NULL,                    // 默认安全级别
        PAGE_READWRITE,          // 可读可写
        0,                       // 高位文件大小
        BUF_SIZE,                // 地位文件大小
        szName                   // 共享内存名称
        );
 
 
    mSemp.pBuf = (char *)MapViewOfFile(
        hMapFile,            // 共享内存的句柄
        FILE_MAP_ALL_ACCESS, // 可读写许可
        0,
        0,
        BUF_SIZE
        );

    mSemp.mFrm = frm;

    AfxBeginThread((AFX_THREADPROC)ThreadProc,(LPVOID)&mSemp,THREAD_PRIORITY_IDLE);
}

Tips :


https://docs.microsoft.com/zh-cn/windows/desktop/api/winbase/nf-winbase-createsemaphorea

https://blog.csdn.net/educast/article/details/8477294
  • 由上图可知,此时已经创建好了共享内存文件(通过CreateFileMapping),并完成映射(通过MapViewOfFile),接下来就是要弄unity里面连接映射的地方了。
  • 2.MFC向共享内存写入数据

//发送消息到u3d
void ProcessMsg::SendMsgToU3D(char *str){
    strncpy(mSemp.pBuf, str, BUF_SIZE);

    //strncpy(mSemp.pBuf, str, BUF_SIZE - 1);
    //mSemp.pBuf[BUF_SIZE - 1] = '\0';

    //放u3d读
    //ReleaseSemaphore(mSemp.hU3DRead,1,0);
    //放u3d开写
    //ReleaseSemaphore(mSemp.hU3DWrite,1,0);
}
  • 3.unity连接映射地址

image.png

Ps:固定格式代码

    const int INVALID_HANDLE_VALUE = -1;
    const int PAGE_READWRITE = 0x04;

    bool _loadingRedStartMap = false;

    //共享内存
    [DllImport("Kernel32.dll", EntryPoint = "CreateFileMapping")]
    private static extern IntPtr CreateFileMapping(IntPtr hFile, //HANDLE hFile,
     UInt32 lpAttributes,//LPSECURITY_ATTRIBUTES lpAttributes,  //0
     UInt32 flProtect,//DWORD flProtect
     UInt32 dwMaximumSizeHigh,//DWORD dwMaximumSizeHigh,
     UInt32 dwMaximumSizeLow,//DWORD dwMaximumSizeLow,
     string lpName//LPCTSTR lpName
     );

    [DllImport("Kernel32.dll", EntryPoint = "OpenFileMapping")]
    private static extern IntPtr OpenFileMapping(
     UInt32 dwDesiredAccess,//DWORD dwDesiredAccess,
     int bInheritHandle,//BOOL bInheritHandle,
     string lpName//LPCTSTR lpName
     );

    const int FILE_MAP_ALL_ACCESS = 0x0002;
    const int FILE_MAP_WRITE = 0x0002;

    [DllImport("Kernel32.dll", EntryPoint = "MapViewOfFile")]
    private static extern IntPtr MapViewOfFile(
     IntPtr hFileMappingObject,//HANDLE hFileMappingObject,
     UInt32 dwDesiredAccess,//DWORD dwDesiredAccess
     UInt32 dwFileOffsetHight,//DWORD dwFileOffsetHigh,
     UInt32 dwFileOffsetLow,//DWORD dwFileOffsetLow,
     UInt32 dwNumberOfBytesToMap//SIZE_T dwNumberOfBytesToMap
     );

    [DllImport("Kernel32.dll", EntryPoint = "UnmapViewOfFile")]
    private static extern int UnmapViewOfFile(IntPtr lpBaseAddress);

    [DllImport("Kernel32.dll", EntryPoint = "CloseHandle")]
    private static extern int CloseHandle(IntPtr hObject);

    [DllImport("Kernel32.dll", EntryPoint = "CreateSemaphoreA")]
    private static extern IntPtr CreateSemaphoreA(
        UInt32 lpSemaphoreAttributes, // SD
        UInt32 lInitialCount, // initial count
        UInt32 lMaximumCount, // maximum count
        string lpName// object name
    );

    [DllImport("Kernel32.dll", EntryPoint = "OpenSemaphoreA")]
    private static extern IntPtr OpenSemaphoreA(
        UInt32 dwDesiredAccess, // access 0x1F0003
        int bInheritHandle, // inheritance option
        string lpName // object name
    );

    [DllImport("Kernel32.dll", EntryPoint = "ReleaseSemaphore")]
    private static extern int ReleaseSemaphore(
        IntPtr hSemaphore,
        UInt32 lReleaseCount,
        Int32 lpPreviousCount
    );

    [DllImport("Kernel32.dll", EntryPoint = "WaitForSingleObject")]
    private static extern UInt32 WaitForSingleObject(
        IntPtr hHandle,
        UInt32 dwMilliseconds
    );

    private IntPtr handle;     //文件句柄
    private IntPtr addr;       //共享内存地址

    IntPtr hMFCWrite;
    IntPtr hU3DRead;

    IntPtr hU3DWrite;
    IntPtr hMFCRead;

    uint mapLength;            //共享内存长

         //  Process pMFC;            //程序
         //  pMFC = Process.Start(Application.streamingAssetsPath + "\\MFC\\MFC_Client.exe");
         //  StartCoroutine(WaitToInit(0.5f));
  • 4.Unity数据读取(注意,使用共享内存时有些地方需要上锁):

image.png
    //不安全的代码在项目生成的选项中选中允许不安全代码
    static unsafe void byteCopy(byte[] dst, IntPtr src)
    {
        fixed (byte* pDst = dst)
        {
            byte* pdst = pDst;
            byte* psrc = (byte*)src;
            while ((*pdst++ = *psrc++) != '\0')
                ;
        }
    }
  • 5.Unity数据发送

image.png
    static unsafe void Copy(byte[] byteSrc, IntPtr dst)
    {
        fixed (byte* pSrc = byteSrc)
        {
            byte* pDst = (byte*)dst;
            byte* psrc = pSrc;
            for (int i = 0; i < byteSrc.Length; i++)
            {
                *pDst = *psrc;
                pDst++;
                psrc++;
            }
        }
    }
  • 5.结束的时候要做点事

1.MFC
image.png

2.Unity

    void OnApplicationQuit()
    {
        if (sw != null)
        {
            sw.Close();
        }

        if (fs != null)
        {
            fs.Close();
        }

        if (isLocalTest)
        {
            return;
        }


        ReleaseSemaphore(hU3DRead, 1, 0);   //不调这一句的话,关闭unity时会卡死

        //清空内存
        byte[] sendStr = Encoding.Default.GetBytes("" + '\0');
        //如果要是超长的话,应另外处理,最好是分配足够的内存
        if (sendStr.Length < mapLength)
        {
            Copy(sendStr, addr);
        }

        //关闭线程
        if (threadRead != null)
        {
            threadRead.Interrupt();
            threadRead.Abort();
        }

        //关闭MFC进程
        if (pMFC != null)
        {
            pMFC.Kill();
        }

        //UnityEngine.Debug.Log("关闭U3D");

        if (addr != null)
        {
            UnmapViewOfFile(addr);
        }

        if (handle != null)
        {
            CloseHandle(handle);
        }

        if (hU3DWrite != null)
        {
            CloseHandle(hU3DWrite);
        }

        if (hMFCRead != null)
        {
            CloseHandle(hMFCRead);
        }

        if (hMFCWrite != null)
        {
            CloseHandle(hMFCWrite);
        }

        if (hU3DRead != null)
        {
            CloseHandle(hU3DRead);
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,093评论 1 32
  • 一、温故而知新 1. 内存不够怎么办 内存简单分配策略的问题地址空间不隔离内存使用效率低程序运行的地址不确定 关于...
    SeanCST阅读 7,784评论 0 27
  • 转载自VR设计云课堂[https://www.jianshu.com/u/c7ffdc4b379e]Unity S...
    水月凡阅读 1,009评论 0 0
  • 最近网友通过网站搜索Unity3D在手机及其他平台下占用内存太大. 这里写下关于unity3d对于内存的管理与优化...
    杨树叶的杨阅读 1,069评论 0 0
  • 文 大冰 是哦,你我皆凡人,哪儿来的那么多永远,比肩过后往往是擦肩。 该来的该去的总会如约发生,就像闪电消失后是倾...
    vivo王大益阅读 307评论 0 0