为了更好地理解这个问题,请引用我之前的问题:

C++ MSAPI 5: SetNotifyCallbackFunction not working

在Microsoft SAPI中,为了在使用SetNotifyCallbackFunction时传递语音转换事件,您需要创建一个消息泵,下面是代码。

现在我的问题是我需要异步完成消息泵。我已经尝试过std::threadpthreadboost库。但是每当我把消息泵放在另一个线程中时。泵出现故障。每当我尝试在另一个线程中调用Speak时,情况也是如此。我该如何解决?同样,我的目标是使MSAPI与事件异​​步说

调用消息泵:

HANDLE hWait = pV->SpeakCompleteEvent();
WaitAndPumpMessagesWithTimeout(hWait, INFINITE);

实际的消息泵代码:
HRESULT WaitAndPumpMessagesWithTimeout(HANDLE hWaitHandle, DWORD dwMilliseconds)
{
    HRESULT hr = S_OK;
    BOOL fContinue = TRUE;

    while (fContinue)
    {
        DWORD dwWaitId = ::MsgWaitForMultipleObjectsEx(1, &hWaitHandle, dwMilliseconds, QS_ALLINPUT, MWMO_INPUTAVAILABLE);
        switch (dwWaitId)
        {
        case WAIT_OBJECT_0:
            {
                fContinue = FALSE;
            }
            break;

        case WAIT_OBJECT_0 + 1:
            {
                MSG Msg;
                while (::PeekMessage(&Msg, NULL, 0, 0, PM_REMOVE))
                {
                    ::TranslateMessage(&Msg);
                    ::DispatchMessage(&Msg);
                }
            }
            break;

        case WAIT_TIMEOUT:
            {
                hr = S_FALSE;
                fContinue = FALSE;
            }
            break;

        default:// Unexpected error
            {
                fContinue = FALSE;
                hr = E_FAIL;
            }
            break;
        }
    }
    return hr;
}

最佳答案

我怀疑您需要在调用WaitAndPumpMessagesWithTimeout之前创建消息队列。

有两种方法可以做到这一点:

  • 调用::PeekMessage(&Msg, NULL, 0, 0, PM_NOREMOVE)
  • 创建一个窗口(仅消息窗口在此处很有用)

  • 在单独的线程上调用Speak()时,也应该在该线程上创建SAPI对象。

    10-07 20:48