我有一个以下列方式启动的com服务器:
extern "C" int WINAPI _tWinMain(HINSTANCE hInstance,
HINSTANCE /*hPrevInstance*/, LPTSTR lpCmdLine, int /*nShowCmd*/)
{
lpCmdLine = GetCommandLine(); //this line necessary for _ATL_MIN_CRT
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
HRESULT hRes = CoInitializeEx(NULL, COINIT_MULTITHREADED);
#else
HRESULT hRes = CoInitialize(NULL);
#endif
_ASSERTE(SUCCEEDED(hRes));
_Module.Init(ObjectMap, hInstance, &LIBID_EXACTACONSOLIDATIONLib);
_Module.dwThreadID = GetCurrentThreadId();
TCHAR szTokens[] = _T("-/");
int nRet = 0;
BOOL bRun = TRUE;
LPCTSTR lpszToken = FindOneOf(lpCmdLine, szTokens);
while (lpszToken != NULL)
{
if (lstrcmpi(lpszToken, _T("UnregServer"))==0)
{
_Module.UpdateRegistryFromResource(IDR_ExactaConsolidation, FALSE);
nRet = _Module.UnregisterServer(TRUE);
bRun = FALSE;
break;
}
if (lstrcmpi(lpszToken, _T("RegServer"))==0)
{
_Module.UpdateRegistryFromResource(IDR_ExactaConsolidation, TRUE);
nRet = _Module.RegisterServer(TRUE);
bRun = FALSE;
break;
}
lpszToken = FindOneOf(lpszToken, szTokens);
}
if (bRun)
{
_Module.StartMonitor();
#if _WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE | REGCLS_SUSPENDED);
_ASSERTE(SUCCEEDED(hRes));
hRes = CoResumeClassObjects();
#else
hRes = _Module.RegisterClassObjects(CLSCTX_LOCAL_SERVER,
REGCLS_MULTIPLEUSE);
#endif
_ASSERTE(SUCCEEDED(hRes));
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);
_Module.RevokeClassObjects();
Sleep(dwPause); //wait for any threads to finish
}
}
我的问题是,如果多个客户端连接到此COM服务器,它是否以同步方式处理请求(对com服务器正在托管的com对象的函数调用)?
我问的原因是由于以下情况: 客户端A 和客户端B 都连接到 Com Server C 。它们都发出请求,并且此请求涉及使用COM对象从数据库中检索下一个项目(在我的情况下为命令)。不应该允许两个客户端具有相同的项目,因此基本上发生的是每个项目在从数据库检索并返回到客户端时被标记为已采用。
如果是2个客户端连接到相同的COM服务器,那么在上面的场景中是否足以将数据库中的项目标记为不必担心竞争条件?如果处理请求同步,我认为将项目标记为已经可以。
答案 0 :(得分:4)
这取决于注册(或者更确切地说,创建)类对象的线程是单线程单元还是多线程单元。请参阅MSDN article on the threading modes。如果您使用CoInitialize(NULL)
或CoInitializeEx(NULL, COINIT_APARTMENTTHREADED)
初始化COM,那么您就在STA中;如果您使用的是CoInitializeEx(NULL, COINIT_MULTITHREADED)
或CoInitializeEx(NULL, 0)
,那么您就是在MTA中。
如果您在MTA中,COM将从线程池同时多次调用您的对象。在STA中,一次只能进行一次呼叫。如果你只是想避免比赛,使用STA可能是一种合理的方法;但请记住,这会产生性能瓶颈,因此对于体积非常大的服务器或请求可能需要一段时间的服务器,使用具有您自己的同步逻辑的MTA可能更为可取。