Сущность технологии COM




Динамическая композиция - часть 3


Каждый раз при получении запроса на интерфейс IBoat размещается новый отделяемый интерфейс. Согласно стандартной практике QueryInterface вызова AddRef посредством результирующего указателя:

((IUnknown*)*ppv)->AddRef();

AddRef будет обрабатывать непосредственно из QueryInterface только отделяемый интерфейс. Важно то, что главный объект остается в памяти столько времени, сколько существует отделяемый интерфейс. Простейший путь обеспечить это — заставить сам отделяемый интерфейс представлять неосвобожденную ссылку. Это можно реализовать в разработчике и деструкторе отделяемого интерфейса:

CarBoat::XBoat::XBoat(CarBoat *pThis) : m_cBoatRef(0), m_pThis(pThis) { m_pThis->AddRef(); }

CarBoat::XBoat::~XBoat(void) { m_pThis->Release(); }

Как и в случае с композицией, методу QueryInterface отделяемого интерфейса требуется идентифицировать объект, делегируя освобождение функции главного объекта. Однако отделяемый интерфейс может выявлять запросы на тот интерфейс (интерфейсы), который он сам реализует, и просто возвращать указатель, обработанный AddRef, себе самому:

STDMETHODIMP CarBoat::XBoat::QueryInterface(REFIID riid, void**ppv) { if (riid != IID_IBoat) return This()->QueryInterface(riid, ppv); *ppv = static_cast<IBoat*>(this); reinterpret_cast<IUnknown*>(*ppv)->AddRef(); return S_OK; }

Ввиду того, что отделяемый интерфейс должен самоуничтожаться, когда он больше не нужен, он должен поддерживать свой собственный счетчик ссылок и уничтожать себя, когда этот счетчик достигнет нуля. Как отмечалось ранее, деструктор отделяемого интерфейса освободит главный объект до того, как сам исчезнет из памяти:

STDMETHODIMP_(ULONG) CarBoat::XBoat::AddRef (void) { return InterlockedIncrement(&m_cRef); }

STDMETHODIMP_(ULONG) CarBoat::X8oat::Release(void) { ULONG res = InterlockedDecrement(&m_cBoatRef); if (res == 0) delete this; // dtor releases main object // деструктор освобождает главный объект return res; }

Как и в случае с композицией, метод This() можно использовать в любых методах отделяемого интерфейса, которым требуется получить статус главного объекта.


Содержание  Назад  Вперед