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



>{

>public:

>  template  // (4)

>  void addReceiver(Address address, CallObject object)

>  {

>    callObjects.insert({ address,object } );

>  }


>  void deleteReceiver(Address address)  // (5)

>  {

>    callObjects.erase(address);

>  }


>  Return operator()(Address address, ArgumentList… arguments)  // (6)

>  {

>    auto iterator = callObjects.find(address);  // (7)

>    if (iterator != callObjects.end())

>    {

>      return iterator->second(arguments…);    // (8)

>    }

>    else

>    {

>      throw std::invalid_argument("Invalid receiver address");  // (9)

>    }

>  }

>private:

>  std::map< Address, std::function, AddressCompare  > callObjects;  // (10)

>};


В строке 1 объявлена общая специализация шаблона, параметрами выступают адрес получателя Address, предикат для сравнения AddressCompare и сигнатура распределяющей функции Function. Реализация здесь отсутствует, поскольку для каждой сигнатуры требуется отдельная специализация – аналогично настройке сигнатуры для универсального аргумента (п. 4.5.2).

В строке 2 объявлена частичная специализация, в которой дополнительно представлены параметр для возвращаемого значения Return и пакет параметров ArgumentList для аргументов функции. В строке 3 объявлен класс, который специализируется сигнатурой из указанных параметров.

В строке 4 объявлен шаблон метода для добавления получателя, который принимает адрес address, вызываемый объект object и добавляет их в контейнер. В строке 5 объявлен метод для удаления получателя. Оба метода работают с контейнером, который объявлен в строке 10. Контейнер объявлен как std::map, ключом является адрес, а значением – объект std::function с заданной сигнатурой.

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

В строке 7 происходит поиск получателя по адресу. Если получатель найден, то происходит вызов объекта (строка 8). Если получатель не найден, то генерируется исключение (строка 9), иначе какой результат нам возвратить?

5.7.3. Использование адресного распределения

Пример использования адресного распределения приведен в Листинг 85.