COM 학습: delete pIUnknown 시 이상 및 IUnknown 순서
#include "stdafx.h"
//
// IUnknown.cpp
// To compile use: cl IUnknown.cpp UUID.lib
//
#include <iostream>
#include <objbase.h>
using namespace std;
void trace(const char* msg) { cout << msg << endl ;}
// Interfaces
interface IX : IUnknown
{
virtual void __stdcall Fx() = 0 ;
} ;
interface IY : IUnknown
{
virtual void __stdcall Fy() = 0 ;
} ;
interface IZ : IUnknown
{
virtual void __stdcall Fz() = 0 ;
} ;
// Forward references for GUIDs
extern const IID IID_IX ;
extern const IID IID_IY ;
extern const IID IID_IZ ;
//
// Component
//
class CA : public IX,
public IY
{
//IUnknown implementation
virtual HRESULT __stdcall QueryInterface(const IID& iid, void** ppv) ;
virtual ULONG __stdcall AddRef() { return 0 ;}
virtual ULONG __stdcall Release() { return 0 ;}
// Interface IX implementation
virtual void __stdcall Fx() { cout << "Fx" << endl ;}
// Interface IY implementation
virtual void __stdcall Fy() { cout << "Fy" << endl ;}
} ;
HRESULT __stdcall CA::QueryInterface(const IID& iid, void** ppv)
{
if (iid == IID_IUnknown)
{
trace("QueryInterface: Return pointer to IUnknown.") ;
*ppv = static_cast<IX*>(this) ;
}
else if (iid == IID_IX)
{
trace("QueryInterface: Return pointer to IX.") ;
*ppv = static_cast<IX*>(this) ;
}
else if (iid == IID_IY)
{
trace("QueryInterface: Return pointer to IY.") ;
*ppv = static_cast<IY*>(this) ;
}
else
{
trace("QueryInterface: Interface not supported.") ;
*ppv = NULL ;
return E_NOINTERFACE ;
}
reinterpret_cast<IUnknown*>(*ppv)->AddRef() ; // See Chapter 4.
return S_OK ;
}
//
// Creation function
//
IUnknown* CreateInstance()
{
IUnknown* pI = static_cast<IY*>(new CA) ;
pI->AddRef() ;
return pI ;
}
//
// IIDs
//
// {32bb8320-b41b-11cf-a6bb-0080c7b2d682}
static const IID IID_IX =
{0x32bb8320, 0xb41b, 0x11cf,
{0xa6, 0xbb, 0x0, 0x80, 0xc7, 0xb2, 0xd6, 0x82}} ;
// {32bb8321-b41b-11cf-a6bb-0080c7b2d682}
static const IID IID_IY =
{0x32bb8321, 0xb41b, 0x11cf,
{0xa6, 0xbb, 0x0, 0x80, 0xc7, 0xb2, 0xd6, 0x82}} ;
// {32bb8322-b41b-11cf-a6bb-0080c7b2d682}
static const IID IID_IZ =
{0x32bb8322, 0xb41b, 0x11cf,
{0xa6, 0xbb, 0x0, 0x80, 0xc7, 0xb2, 0xd6, 0x82}} ;
//
// Client
//
int _tmain(int argc, _TCHAR* argv[])
{
HRESULT hr ;
trace("Client: Get an IUnknown pointer.") ;
IUnknown* pIUnknown = CreateInstance() ;
trace("Client: Get interface IX.") ;
IX* pIX = NULL ;
hr = pIUnknown->QueryInterface(IID_IX, (void**)&pIX) ;
if (SUCCEEDED(hr))
{
trace("Client: Succeeded getting IX.") ;
pIX->Fx() ; // Use interface IX.
}
trace("Client: Get interface IY.") ;
IY* pIY = NULL ;
hr = pIUnknown->QueryInterface(IID_IY, (void**)&pIY) ;
if (SUCCEEDED(hr))
{
trace("Client: Succeeded getting IY.") ;
pIY->Fy() ; // Use interface IY.
}
trace("Client: Ask for an unsupported interface.") ;
IZ* pIZ = NULL ;
hr = pIUnknown->QueryInterface(IID_IZ, (void**)&pIZ) ;
if (SUCCEEDED(hr))
{
trace("Client: Succeeded in getting interface IZ.") ;
pIZ->Fz() ;
}
else
{
trace("Client: Could not get interface IZ.") ;
}
trace("Client: Get interface IY from interface IX.") ;
IY* pIYfromIX = NULL ;
hr = pIX->QueryInterface(IID_IY, (void**)&pIYfromIX) ;
if (SUCCEEDED(hr))
{
trace("Client: Succeeded getting IY.") ;
pIYfromIX->Fy() ;
}
trace("Client: Get interface IUnknown from IY.") ;
IUnknown* pIUnknownFromIY = NULL ;
hr = pIY->QueryInterface(IID_IUnknown, (void**)&pIUnknownFromIY) ;
if (SUCCEEDED(hr))
{
cout << "Are the IUnknown pointers equal? " ;
if (pIUnknownFromIY == pIUnknown)
{
cout << "Yes, pIUnknownFromIY == pIUnknown." << endl ;
}
else
{
cout << "No, pIUnknownFromIY != pIUnknown." << endl ;
}
}
// Delete the component.
delete pIUnknown ;
//pIUnknown->Release();
return 0 ;
}
위의 예는 실행하기에 아무런 문제가 없지만,Create Instance () 를IUnknown* CreateInstance()
{
IUnknown* pI = static_cast<IY*>(new CA) ;
pI->AddRef() ;
return pI ;
} 어떤 영향이 있습니까?분석상으로는 아무런 문제가 없다. 단지 여러 계승의 대상을 어느 부류의 대상으로 강등시켰을 뿐이다. 그러나 delete pIUnknown으로 실행할 때 ASSERT 이상이 발생하면 F10이 발견된다ASSERTE(_BLOCK_TYPE_IS_VALID(pHead->nBlockUse));실수
이 때 pIUnknown 바늘은 당초에 분배된 전체 메모리 블록을 가리키지 않기 때문에 new에서 메모리에 헤더 정보를 씁니다. 이 때 pIUnknown에서 헤더 정보를 찾을 수 없어 delete가 실패합니다.
delete에 성공하려면 상속 순서를 수정해야 합니다.
class CA : public IY, public IX{...}
물론 여기에는 pIUnknown->Release()라면 어떤 결과가 나올지 의문이 하나 더 있다.
이 예에서 Release () 함수는 아무런 처리도 하지 않았기 때문에 메모리 유출을 초래할 수 있습니다. Release () 안에도 delete가 있다면, 강전의 유형이 정말 중요합니다.
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
다양한 언어의 JSONJSON은 Javascript 표기법을 사용하여 데이터 구조를 레이아웃하는 데이터 형식입니다. 그러나 Javascript가 코드에서 이러한 구조를 나타낼 수 있는 유일한 언어는 아닙니다. 저는 일반적으로 '객체'{}...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.