Обратные вызовы в C++ | страница 55



Листинг 59. Использование инициатора с оболочкой std::function для вызова методов класса

>class Executor

>{

>public:

>  static void staticCallbackHandler(Executor* executor, int eventID) {}

>  void callbackHandler(int eventID) {}

>  void operator() (int eventID) {}

>};


>int main()

>{

>  Executor executor;


>  Initiator initiator;  // (1)

>  initiator.setup(CallbackConverter(&Executor::callbackHandler, &executor));  // (2)

>  initiator.run();


>  Initiator initiatorForClass;                // (3)

>  initiatorForClass.setup(&Executor::callbackHandler);  // (4)

>  initiatorForClass.setupInstance(&executor);           // (5)

>  initiatorForClass.run();

>}


В строке 1 объявлен исходный инициатор. В параметры шаблона мы не передаем никаких аргументов, т. е. шаблон инстанциируется подобно обычному классу. В строке 2 происходит настройка инициатора, в качестве аргумента передается объект для преобразования вызовов.

В строке 3 объявлен специализированный инициатор для вызова методов класса, он инстанциируется типом Executor. В строке 4 настраивается указатель на метод класса, в строке 5 настраивается указатель на экземпляр класса.


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

4.6.6. Перенаправление вызовов

Представьте следующую ситуацию: инициатор вызывает функцию с одной сигнатурой, а в клиенте реализован обработчик с другой сигнатурой. Например, в исполнителе реализована функция обработки нажатия кнопки, которая на вход принимает два параметра – идентификатор кнопки и текущее поле редактирования. В то же время инициатор вызывает функцию, передавая ей только один аргумент – идентификатор текущей нажатой кнопки, и он ничего не знает об остальных элементах управления. Можно ли сделать так, чтобы инициатор вызывал одну функцию, но при этом бы вызывалась другая функция, другими словами, происходило перенаправление вызова? В стандартной библиотеке для этого существуют специальные объекты связывания