进程间同步(互斥) 以及 同一DLL的不同进程copy共享全局变量

来源:互联网 发布:红外光谱数据库 编辑:程序博客网 时间:2024/05/19 12:38

     此前遇到一个问题,希望实现一个COM组件,能够被不同的AP调用,同时需要他们保持同步。比较好的方式是采用ATL实现singleton的Com组件,很遗憾,这部分并不熟悉,只是依照codeproject上的sample"葫芦"画了一个瓢。看起来是可以用,大概的流程也清楚,但是说起ATL,根本没看过。

     那好吧,既然互斥就可以了,那就挑简单的,实现一个普通的COM组件/DLL,利用进程间互斥来实现他们的同步。进程间互斥的方法很多,包括内存文件映射、信号量等,通常用的是信号量,信号量的种类很多,譬如mutex,event,criticalsection等等。这里以mutex为例,简单实现了进程间互斥。

    互斥的核心思想是创建一个共享的信号量,只能被一个进程独占,而在已经被一个进程独占的时候,其他进程需要等待此信号量,直到占用此信号量的进程释放他。进程间之所以能够共享某一信号量,主要是因为信号量是一个内核对象,在进程中看到的是他在某个进程空间中的一个映射handle,在不同的进程空间映射的handle可能不同,但是在内核空间他们是同一个对象,也就是通过不同进程空间的不同handle去操作同一个内核对象。

    以mutex为例,可以通过CreateMutex来创建一个有名字的mutex的handle,如果此前没有其他程序创建过,系统会创建此mutex,并返回handle;如果此前已经有其他程序创建过,系统会返回一个handle,handle对应的内核对象是上次已经创建了的。此时如果利用GetLastError()来获取error code会得到ERROR_ALREADY_EXISTS,利用此特性,很多人用这种方法来限制只运行一个实例。譬如在程序启动时,通过创建特殊名字的mutex来确定之前是否已经启动过此程序(CreateMutex之后调用GetLastError是否返回ERROR_ALREADY_EXISTS),code如下:

m_mutexHandle = CreateMutex(NULL, FALSE, _T("XXxxXX"));if(m_mutexHandle==NULL){ return FALSE;}if (GetLastError()   ==   ERROR_ALREADY_EXISTS){CloseHandle(m_mutexHandle);m_mutexHandle = NULL;CString szInfo;AfxMessageBox(_T("ALREADY_EXISTS"));return FALSE;}

 

    这里不作过多讨论,看如何实现进程间互斥。

1,创建mutex

HANDLE m_hMutex = CreateMutex(NULL, FALSE, L"RtsKeyStone_Mutex");

2,利用此mutex实现进程同步/函数互斥访问

void Func(){WaitForSingleObject(m_hMutex, INFINITE);// to do...ReleaseMutex(m_hMutex);return;}

这样可以保证两个进程不可同时访问func()的实现.

    OK,进程间同步实现了,但是如果是DLL中的全局变量,如何共享呢?网上推荐在DLL中定义共享数据段。试了下,此方法确实可以凑效。#pragma data_seg预处理指令用于设置共享数据段。

#progma data_set("SharedDataName")int g_VendorPrevNum = 0; // need to init#pragma data_seg()#pragma comment(linker, "/Section:SharedDataName,RWS")

在#pragma data_set("SharedDataName")和#pragma data_set()之间的所有变量将被访问该DLL的所有进程看到、共享。

具体细节可以参考:http://blog.csdn.net/xuplus/article/details/2291860

 

原创粉丝点击