我有 MFC 应用程序,它使用并行模式库来执行一些异步任务。其中一些使用 COM 对象,因此我需要在此类任务中初始化 COM 库。在所有这些情况下,我都使用 COM STA 模型初始化,因为主线程是 MFC 应用程序(MFC 应用程序线程只能是 STA)并且我不知道我的任务将在哪个线程上下文中被调用。
一些例子:
BOOL CMyApp::InitInstance() {
// base initialization
CWinAppEx::InitInstance();
AfxOleInit();
// ... some code ...
// PPL usage
{
Concurrency::task_group aTasks;
// Task1
aTasks.run([&](){
HRESULT hRes = ::CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
if (SUCCEEDED(hRes)) {
Sleep(100);
::CoUninitialize();
}
});
// Task2
aTasks.run([&](){
HRESULT hRes = ::CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
if (SUCCEEDED(hRes)) {
Sleep(100);
::CoUninitialize();
}
});
// Task3
aTasks.run([&](){
HRESULT hRes = ::CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
if (SUCCEEDED(hRes)) {
Sleep(100);
::CoUninitialize();
}
});
aTasks.wait();
}
}
此代码在 Windows 7/XP 上运行正常。但是在带有 C++ 2012 Platform Toolset 任务 1 和 2 的 Windows 8.1 上无法正常工作,因为 CoInitializeEx() 返回 RPC_E_CHANGED_MODE 错误!任务 3 通常由 PPL 核心在主 MFC 线程上下文中调用,该线程上下文是 OLE,并且他的 COM 已经初始化为 COINIT_APARTMENTTHREADED,因此 CoInitializeEx() 返回成功 S_FALSE 代码(双重初始化)。
对于任务 2 和 3,PPL 核心创建单独的线程,这些线程在 Windows 7/XP 上未预初始化为 COM,因此任务第一行成功初始化 COM。 但是在 Windows 8.1 上,所有线程看起来都被预初始化为带有 COINIT_MULTITHREADED 标志的 COM,随后的 CoInitializeEx(..., COINIT_APARTMENTTHREADED) 调用返回错误!
我勒个去!如何在 Window 8.1 上定义正确的 COM 初始化规则?我的错误在哪里?PPL 不能保证我的任务线程上下文,它可以是 MFC 中必须是 STA 的主线程。而且我无法定义何时应该使用 MTA 或 STA COM 初始化。
请帮我。可能这是 2012 C++ 平台工具集的 PPL 核心代码中的错误,还是 Windows 8.1 中的 PPL 使用错误?