Делегаты на C++ | страница 3



> CMethodDelegateVoid‹TObj›* pMethodDel = dynamic_cast‹CMethodDelegateVoid‹TObj›* ›(pDelegate);

> if (pMethodDel == NULL || pMethodDel-›m_pObj != m_pObj || pMethodDel-›m_pMethod != m_pMethod) return false;

> return true;

>}


Классы CStaticDelegateVoid и CMethodDelegateVoid можно использовать непосредственно. Но для пользователя удобнее работать исключительно с интерфейсом IDelegateVoid, не задумываясь о существовании двух различных классов реализации. Поэтому напишем перегруженную функцию NewDelegate, которая будет создавать нужный объект и возвращать пользователю IDelegateVoid*. Её реализация будет выглядеть так:

>IDelegateVoid* NewDelegate(void (*pFunc)()) {

> return new CStaticDelegateVoid(pFunc);

>}


>template ‹class TObj›

>IDelegateVoid* NewDelegate(TObj* pObj, void (TObj::*pMethod)()) {

> return new CMethodDelegateVoid‹TObj› (pObj, pMethod);

>}


Мы уже почти закончили. Осталось написать объектную обёртку над интерфейсом IDelegateVoid, которая будет поддерживать список указателей и определять набор операторов, аналогичных используемым в C# - operator=, operator(), operator+= и operator-=. Для простоты будем использовать стандартный класс std::list для хранения списка указателей.

>#include ‹list›


>class CDelegateVoid {

>public:

> CDelegateVoid(IDelegateVoid* pDelegate = NULL) {

> Add(pDelegate);

>}

> ~CDelegateVoid() { RemoveAll(); }

> bool IsNull() { return (m_DelegateList.size() == 0); }

> CDelegateVoid& operator=(IDelegateVoid* pDelegate) {

>  RemoveAll();

>  Add(pDelegate);

>  return *this;

> }

> CDelegateVoid& operator+=(IDelegateVoid* pDelegate) {

>  Add(pDelegate);

>  return *this;

> }

> CDelegateVoid& operator-=(IDelegateVoid* pDelegate) {

>  Remove(pDelegate);

>  return *this;

> }

> void operator()() { Invoke(); }

>private:

> void Add(IDelegateVoid* pDelegate);

> void Remove(IDelegateVoid* pDelegate);

> void RemoveAll();

> void Invoke();

>private:

> std::list‹IDelegateVoid*› m_DelegateList;

>};


Для реализации необходимого набора операторов используются вспомогательные методы Add, Remove, RemoveAll и Invoke. Метод Add добавляет новый указатель IDelegateVoid* в список:

>void CDelegateVoid::Add(IDelegateVoid* pDelegate) {

> if (pDelegate != NULL) m_DelegateList.push_back(pDelegate);

>}


Метод Remove ищет в списке делегат, ссылающийся на заданную функцию, и в случае обнаружения удаляет его:

>void CDelegateVoid::Remove(IDelegateVoid* pDelegate) {

> std::list‹IDelegateVoid*›::iterator it;

> for(it = m_DelegateList.begin(); it!= m_DelegateList.end(); ++it) {