【发布时间】:2016-06-13 16:56:13
【问题描述】:
我正在调试客户网站上的间歇性问题。我已经明白了,对 COM 对象的Release() 调用似乎没有返回。
第一个日志正在打印,但我从未看到第二个日志。我只能假设对Release() 的调用由于某种原因从未返回(或者可能是CoInitializeEx())。
我不知道接下来要寻找什么,任何帮助/线索将不胜感激。
Logger::getLogger()->logTrace("AudioCapturer::_shutdown. _pEndpointAudioClient_COM Released. (%s)", _deviceName.c_str());
releaseComObject(_pAudioCaptureClient_COM);
Logger::getLogger()->logTrace("AudioCapturer::_shutdown (%s) succeeded", _deviceName.c_str());
这里是支持代码:
IAudioCaptureClient *_pAudioCaptureClient_COM;
// Class that Initializes COM on creation and Unitializes on destruction
AutoCOM::AutoCOM() { CoInitializeEx(NULL, COINIT_MULTITHREADED); }
AutoCOM::~AutoCOM() { CoUninitialize(); }
#define AUTO_COM_SUPPORT AutoCOM _autoCOM
// Safe releasing of COM objects. Zeros the pointer
// to the COM object. Safe to use with NULL
// pointers.
template <class T> void releaseComObject(T*& pT) {
if (pT) {
AUTO_COM_SUPPORT;
(pT)->Release();
pT = NULL;
}
}
【问题讨论】:
-
MSDN 文档对 IAudioCaptureClient::Release() 有一个非常具体的警告。引用:“当释放 IAudioCaptureClient 接口实例时,客户端必须从与创建对象的 IAudioClient::GetService 调用相同的线程中调用实例的 Release 方法。”违反这样的要求很可能会导致死锁。
-
如果
CoInitializeEx调用失败,你仍然在做CoUnitialize,这是错误的。 -
#define AUTO_COM_SUPPORT AutoCOM哇,你不想那样做。releaseComObject应该知道 COM 已经初始化或者pT来自哪里?把AUTO_COM_SUPPORT拿出来烧掉。 -
@HansPassant,谢谢。看起来
Release()是从不同的线程调用的。我会修复它,看看是否有帮助。 -
真的,AutoCom 宏需要在火灾中死掉。