글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
COM 개체로 인해 IE 7 비스타 버전이 종료될 때 오류 화면이 뜬다면?
My Toolbar or BHO is Causing IE7 on Vista to Crash on Close. Help!
; https://docs.microsoft.com/en-us/archive/blogs/tonyschr/my-toolbar-or-bho-is-causing-ie7-on-vista-to-crash-on-close-help
코딩에서도, "형식 안정성"을 중요하게 여기는 것처럼, 점차로 마이크로소프트의 전반적인 제품군에서 "오류임에도 불구하고 덮어주던 방식"을 없애는 방향으로 가고 있는 것 같습니다. 위의 기사에서 언급하는 문제도 그와 같은 식이라고 볼 수 있겠지요.
CoInitialize() / CoUninitialize()의 쌍이 맞지 않는 경우에 IE 6까지는 그러한 실수를 감안하여 문제가 없도록 되어 있었는데, IE 7부터는 문제가 발생하게 되었다고 합니다. 만약, 여러분들의 프로그램에서 - 사실 별도의 스레드를 만들지 않는 한 CoInitialize() / CoUninitialize()를 호출할 일은 거의 없지만, 만약 사용하고 있다면 반드시 그 부분에 대한 호출 쌍이 맞는지 확인하셔야 할 것입니다.
그러면서, "IInitializeSpy" 인터페이스를 소개해 주고 있습니다. 어허... 제가 그동안 많이 무심했습니다. 이런 인터페이스가 있었는 줄 처음 알았으니까요. (Windows XP SP1부터 지원되었다고 합니다.) 이 인터페이스는 CoInitialize / CoUninitialize 메서드들이 호출될 때마다 Pre/Post 콜백 함수를 불려지는 것을 목적으로 정의된 것입니다.
테스트 삼아서 간단한 예제를 한번 작성해 보았습니다. (참고로, 위의 기사를 쓴 사람은 절대로 ActiveX 개발자들이 자신들의 CoInitialize() / CoUninitialize() 호출 쌍을 보정하기 위해 이 인터페이스를 구현하지 말라고 당부하고 있습니다.)
참고로, 아래의 예제에 대해 컴파일 가능한 VC++ 8 프로젝트파일은 첨부파일에 넣어두었습니다.
#include "stdafx.h"
#include <windows.h>
#include <objbase.h>
#include <objidl.h>
#pragma comment(lib, "ole32.lib")
class CInitializeSpy : public IInitializeSpy
{
public:
CInitializeSpy()
{
dwCount = 0;
}
STDMETHOD(PreInitialize)(DWORD dwCoInit,DWORD dwCurThreadAptRefs)
{
::OutputDebugStr(L"PreInitialize\r\n");
return S_OK;
}
STDMETHOD(PostInitialize)(HRESULT hrCoInit,DWORD dwCoInit,DWORD dwNewThreadAptRefs)
{
::OutputDebugStr(L"PostInitialize\r\n");
return S_OK;
}
STDMETHOD(PreUninitialize)(DWORD dwCurThreadAptRefs)
{
::OutputDebugStr(L"PreUninitialize\r\n");
return S_OK;
}
STDMETHOD(PostUninitialize)(DWORD dwNewThreadAptRefs)
{
::OutputDebugStr(L"PostUninitialize\r\n");
return S_OK;
}
STDMETHOD(QueryInterface)(REFIID riid, void **ppvObject)
{
bool qied = false;
if ( riid == IID_IUnknown)
{
*ppvObject = this;
qied = true;
}
if ( riid == IID_IInitializeSpy)
{
*ppvObject = this;
qied = true;
}
if ( qied == true )
{
dwCount ++;
return S_OK;
}
return E_NOINTERFACE;
}
virtual ULONG STDMETHODCALLTYPE AddRef( void)
{
dwCount ++;
return dwCount;
}
virtual ULONG STDMETHODCALLTYPE Release( void)
{
dwCount --;
if ( dwCount == 0 )
{
delete this;
}
return dwCount;
}
DWORD dwCount;
};
int _tmain(int argc, _TCHAR* argv[])
{
CInitializeSpy *pSpy = new CInitializeSpy();
ULARGE_INTEGER ulCookie;
CoRegisterInitializeSpy(pSpy, &ulCookie);
CoInitialize(NULL);
CoUninitialize();
CoRevokeInitializeSpy(ulCookie);
return 0;
}
[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]
... [181] 182 183 184 185 186 187 188 189 190 191 192 193 194 195 ...
... [181] 182 183 184 185 186 187 188 189 190 191 192 193 194 195 ...