具有多个客户端的 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 是不是是多线程连接的?的主要内容,如果未能解决你的问题,请参考以下文章

java中具有多个客户端的网络服务器

是否有具有多个命名 Oidc 客户端的示例 quarkus 项目来调用多个服务调用?

c - 具有多个客户端的 UDP 客户端-服务器

具有多个客户端的 Obj-C 服务器

如何将实体框架用于具有多个客户端的应用程序

C ++ - 具有多个客户端的UDP服务器