Делегаты на C++

Делегаты на C++

Авторы:

Жанр: Программирование

Циклы: не входит в цикл

Формат: Полный

Всего в книге 7 страниц. Год издания книги - 2003.

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность. Книга завершается финалом, связывающим воедино темы и сюжетные линии, исследуемые на протяжении всей истории. В целом, книга представляет собой увлекательное и наводящее на размышления чтение, которое исследует человеческий опыт уникальным и осмысленным образом.

Читать онлайн Делегаты на C++


Введение

Делегаты - это объектно-ориентированные указатели на функции, используемые для callback-вызовов в среде CLR фирмы Microsoft. Делегат можно связать со статической функцией или с нестатическим методом любого класса (единственное условие - совпадение сигнатуры метода с сигнатурой, указанной в описании делегата). Затем связанную с делегатом функцию или метод можно вызывать, используя стандартный синтаксис вызова функции в C++. Несколько делегатов можно связать в цепочку. Благодаря этому можно "одним махом" вызвать все связанные с ними callback-функции. Следующий пример демонстрирует применение делегатов в языке C#.

>using System;

>using System.IO;


>namespace CSharpDelegates {

> class App {

>  // Определяем делегат Callback,

>  // который принимает 1 параметр и ничего не возвращает.

>  public delegate void Callback(string str);


>  // Это метод класса App.

>  public void OutputToConsole(string str) {

>   Console.WriteLine(str);

>  }


>  // А это статический метод класса App.

>  public static void OutputToFile(string str) {

>   StreamWriter sw = new StreamWriter("output.txt", true);

>   sw.WriteLine(str);

>   sw.Close();

>  }


>  public static void Main(string[] args) {

>   App app = new App();


>   // Создаём делегат.

>   App.Callback callback = null;

>   if (callback != null) callback("1");


>   // Добавляем ссылку на OutputToFile.

>   // Вызываем её через делегата.

>   callback += new App.Callback(App.OutputToFile);

>   if (callback != null) callback("2");


>   // Добавляем ссылку на OutputToConsole.

>   // Вызывается вся цепочка:

>   // сначала OutputToFile, потом OutputToConsole.

>   callback += new App.Callback(app.OutputToConsole);

>   if (callback != null) callback("3");


>   // Убираем ссылку на OutputToFile.

>   // Вызывается только OutputToConsole.

>   callback -= new App.Callback(App.OutputToFile);

>   if (callback!= null) callback("4");


>   // Убираем оставшуюся ссылку на OutputToConsole.

>   callback -= new App.Callback(app.OutputToConsole);

>   if (callback != null) callback("5");

>  }

> }

>}


Делегаты в CLR удобны, типобезопасны и эффективны. Последнее время на форумах RSDN часто поднимается вопрос о том, можно ли реализовать делегаты с аналогичными свойствами, оставаясь в рамках "чистого" C++. Оказывается, это вполне возможно. В этой статье я покажу, как это сделать.

Частное решение

Для начала создадим делегат для callback-вызова функций и методов с простейшей сигнатурой void(void). Интерфейс этого делегата будет выглядеть так.

>class IDelegateVoid {

>public:

> virtual ~IDelegateVoid() {}

> virtual void Invoke() = 0;

> virtual bool Compare(IDelegateVoid* pDelegate) = 0;

>};


Invoke используется для вызова функции или метода, связанного с делегатом, а Compare сравнивает 2 делегата и возвращает true, если они связаны с одной и той же функцией (методом). Очевидно, что реализация интерфейса IDelegateVoid будет отличаться для статических функций и нестатических методов класса, поэтому мы разнесём эти реализации по различным классам. Класс CStaticDelegateVoid будет "отвечать" за статические функции, а класс CMethodDelegateVoid - за нестатические методы.

Класс CStaticDelegateVoid просто инкапсулирует указатель типа void (*)():

>class CStaticDelegateVoid: public IDelegateVoid {

>public:

> typedef void (*PFunc)();

> CStaticDelegateVoid(PFunc pFunc) { m_pFunc = pFunc; }

> virtual void Invoke() { m_pFunc(); }

> virtual bool Compare(IDelegateVoid* pDelegate);

>private:

> PFunc m_pFunc;

>};


Метод Compare должен проверить, что переданный ему указатель IDelegateVoid* в действительности ссылается на объект CStaticDelegateVoid. Если это не так, делегаты различны (ссылаются на разные функции) и Compare просто возвращает false. Иначе результат определяется сравнением переменных-членов m_pFunc у двух объектов. Реализация этой идеи выглядит так.

>bool CStaticDelegateVoid::Compare(IDelegateVoid *pDelegate) {

> CStaticDelegateVoid* pStaticDel = dynamic_cast‹CStaticDelegateVoid*›(pDelegate);

> if (pStaticDel == NULL || pStaticDel-›m_pFunc != m_pFunc) return false;

> return true;

>}


Класс CMethodDelegateVoid чуть-чуть сложнее. Он должен инкапсулировать указатель на объект и указатель на метод этого объекта. Поскольку в C++ указатели на методы двух разных классов принципиально отличаются (и могут даже иметь разный размер), нам нужна отдельная реализация CMethodDelegateVoid для каждого нового класса, на методы которого мы хотим ссылаться. Поэтому класс CMethodDelegateVoid должен быть шаблоном. В остальном его реализация аналогична CStaticDelegateVoid.

>template‹class TObj›

>class CMethodDelegateVoid: public IDelegateVoid {

>public:

> typedef void (TObj::*PMethod)();

> CMethodDelegateVoid(TObj* pObj, PMethod pMethod) {

>  m_pObj = pObj;

>  m_pMethod = pMethod;

> }

> virtual void Invoke() {

> (m_pObj-›*m_pMethod)();

> }

> virtual bool Compare(IDelegateVoid* pDelegate);

>private:

> TObj *m_pObj;

> PMethod m_pMethod;

>};


>template‹class TObj›

>bool CMethodDelegateVoid‹TObj›::Compare(IDelegateVoid* pDelegate) {

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

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


С этой книгой читают
Питон — модули, пакеты, классы, экземпляры

Python - объектно-ориентированный язык сверхвысокого уровня. Python, в отличии от Java, не требует исключительно объектной ориентированности, но классы в Python так просто изучить и так удобно использовать, что даже новые и неискушенные пользователи быстро переходят на ОО-подход.


Обработка событий в С++

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.


Симуляция частичной специализации
Автор: П Кузнецов

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.


MFC и OpenGL

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.


Java 7

Рассмотрено все необходимое для разработки, компиляции, отладки и запуска приложений Java. Изложены практические приемы использования как традиционных, так и новейших конструкций объектно-ориентированного языка Java, графической библиотеки классов Swing, расширенной библиотеки Java 2D, работа со звуком, печать, способы русификации программ. Приведено полное описание нововведений Java SE 7: двоичная запись чисел, строковые варианты разветвлений, "ромбовидный оператор", NIO2, новые средства многопоточности и др.


Pro Git

Разработчику часто требуется много сторонних инструментов, чтобы создавать и поддерживать проект. Система Git — один из таких инструментов и используется для контроля промежуточных версий вашего приложения, позволяя вам исправлять ошибки, откатывать к старой версии, разрабатывать проект в команде и сливать его потом. В книге вы узнаете об основах работы с Git: установка, ключевые команды, gitHub и многое другое.В книге рассматриваются следующие темы:основы Git;ветвление в Git;Git на сервере;распределённый Git;GitHub;инструменты Git;настройка Git;Git и другие системы контроля версий.


Принц Отто

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.


Маркхейм

В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.


История России с древнейших времен до конца XVII века

"История России с древнейших времен до конца XVII века" - уникальное учебное пособие, созданное коллективом авторов под редакцией Л.В.Милова - известного историка, академика РАН, лауреата Государственной премии РФ. В издании предпринята попытка проследить воздействие природно-климатического и географического факторов на протяжении столетий не только на социально-экономическую жизнь страны, но и их влияние на государственно-политический строй, внутреннюю и внешнюю политику, культуру.Авторы этой книги - ведущие преподаватели Исторического факультета МГУ им.


История России XVIII-XIX веков

"История России XVIII-XIX веков" - уникальное учебное пособие, созданное на основе новейших концепций российского исторического процесса. В издании предпринята попытка проследить воздействие природно-климатического и географического факторов не только на социально-экономическую жизнь страны, но и их влияние на государственно-политический строй, внутреннюю и внешнюю политику, культуру. Авторы этой книги - ведущие преподаватели исторического факультета МГУ им. М.В.Ломоносова - академик РАН, лауреат Государственной премии РФ профессор Л.В.Милов и доктор исторических наук, профессор Н.И.Цимбаев - обобщили свой многолетний опыт исследования и преподавания истории России.