具有多个客户端的 Windows Com Server 是不是是多线程连接的?
Posted
技术标签:
【中文标题】具有多个客户端的 Windows Com Server 是不是是多线程连接的?【英文标题】:Is Windows Com Server with multiple clients connected multi threaded?具有多个客户端的 Windows Com Server 是否是多线程连接的? 【发布时间】:2011-09-30 19:23:38 【问题描述】:我有一个通过以下方式启动的 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 服务器 C。他们都发出一个请求,这个请求涉及使用 COM 对象从数据库中检索下一个项目(在我的例子中是订单)。不应允许两个客户端拥有相同的项目,因此基本上发生的情况是,当从数据库中检索并返回给客户端时,每个项目都被标记为已使用。
如果有 2 个客户端连接到 同一个 COM 服务器,在上面的场景中将数据库中的项目标记为已使用就足够了,而不必担心 竞争条件?如果请求是同步处理,我认为只需将项目标记为已使用就可以了。
【问题讨论】:
【参考方案1】:这取决于注册(或者更确切地说,创建)类对象的线程是单线程单元还是多线程单元。请参阅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 可能更可取。
【讨论】:
+1 非常好的观点。我已将缺少的代码添加到上面的等式中。当我调试 (Win XP) 时,它在 STA (CoInitialize(NULL);) 中。 @Cole,所以如果_WIN32_WINNT >= 0x0400 & defined(_ATL_FREE_THREADED)
,你就在 MTA 中。所以你需要要么自己同步,要么强制进入 STA
在这里感谢您的帮助。我想你已经为我解决了这个问题。以上是关于具有多个客户端的 Windows Com Server 是不是是多线程连接的?的主要内容,如果未能解决你的问题,请参考以下文章