Обратные вызовы в C++ - [47]
> private:
> SensorContainer* sensorContainer_; // (1)
> CommandQueue* commandQueue_; // (2)
> AlertControl* alertControl_; // (3)
> bool isInitialized_; // (4)
> DriverPointer driver_; // (5)
> void checkInitialize(); // (6)
> void checkDriver(); // (7)
>};
>}; //namespace sensor
В строке 1 объявлен контейнер для хранения датчиков, в строке 2 – класс для выполнения асинхронных запросов, в строке 3 – класс для отслеживания пороговых значений. Соответствующие указатели создаются в конструкторе и уничтожаются в деструкторе. Индикатор 4 указывает, была ли выполнена инициализация.
В строке 6 объявлен вспомогательный метод, который проверяет, была ли выполнена инициализация (если нет, выбрасывает исключение). В строке 7 аналогичный метод проверяет, был ли установлен драйвер.
Рассмотрим, как здесь используются обратные вызовы. Для начала самый простой случай – чтение показаний работоспособных датчиков (Листинг 99).
>void SensorControl::readSensorValues(SensorValueCallback callback)
>{
> checkInitialize(); // (1)
> sensorContainer_->forEachSensor([callback](SensorNumber number, SensorPointer sensor) // (2)
> {
> if (sensor->isOperable()) // (3)
> {
> callback(number, sensor->getValue()); // (4)
> }
> }
> );
>}
В строке 1 производится проверка, инициализирован ли класс. Если класс не проинициализирован, то функция выбросит исключение.
В строке 2 происходит перебор элементов контейнера, в качестве обратного вызова используется лямбда-выражение. Контейнер будет вызывать лямбда-выражение, в которое он будет передавать номер датчика и указатель на экземпляр класса. В теле выражения проверяется, является ли датчик работоспособным (строка 3), и если да, то выполняется соответствующий обратный вызов (строка 4).
Рассмотрим теперь поиск максимального и минимального значения для заданного диапазона номеров датчиков. Вначале разработаем вспомогательный класс, который будет последовательно принимать на вход показания датчиков и искать среди них максимальное и минимальное значение (Листинг 100).
>class FindMinMaxValue
>{
>public:
> enum MinMaxSign { MIN_VALUE = 0, MAX_VALUE = 1 }; // (1)
> FindMinMaxValue(SensorNumber first, SensorNumber last, MinMaxSign sign) : // (2)
> sign_(sign), first_(first), last_(last), count_(0)
> {
> if (sign == MIN_VALUE)
> {
> result_ = std::numeric_limits
> }
> else
> {
> result_ = std::numeric_limits
> }
> arrayFunMinMax_[MIN_VALUE] = &FindMinMaxValue::CompareMin; // (5)
> arrayFunMinMax_[MAX_VALUE] = &FindMinMaxValue::CompareMax; // (6)
> }
> void operator()(SensorNumber number, SensorPointer sensor) // (7)
> {
> if ( sensor->isOperable() && (number >= first_ && number <= last_) ) // (8)
> {
> (this->*arrayFunMinMax_[sign_])(sensor->getValue()); // (9)
> count_++; // (10)
> }
> }
> SensorValue result() { return result_; } // (11)
> size_t count() { return count_; } // (12)
>private:
> SensorNumber first; // (13)
> SensorNumber last; // (14)
> MinMaxSign sign; // (15)
> SensorValue result; // (16)
> size_t count; // (17)
> using FunMinMax = void (FindMinMaxValue::*)(SensorValue value); // (18)
> void CompareMin(SensorValue value) // (19)
> {
> if (result_ > value)
> {
> result_ = value;
> }
> }
> void CompareMax(SensorValue value) // (20)
> {
> if (result_ < value)
> {
> result_ = value;
> }
> }
> FunMinMax arrayFunMinMax_[2]; // (21)
>};
В строке 2 объявлен конструктор, который принимает на вход следующие параметры: минимальное значение диапазона номеров; максимальное значение диапазона номеров; параметр, указывающий, что необходим поиск минимального либо максимального значения. В конструкторе инициализируются переменные класса: минимальное значение диапазона (объявлено в строке 13); максимальное значение диапазона (объявлено в 14); параметр для поиска (объявлено в 15); итоговый результат (объявлено в 16); количество датчиков, которые участвовали в поиске (объявлено в 17). В зависимости от переданного параметра начальный результат инициализируется соответственно максимальным либо минимальным значением (строки 3 и 4). Кроме того, инициализируется массив указателей на функцию (строки 5 и 6, объявление в 21). Данные функции предназначены для сравнения и запоминания максимального либо минимального значений (объявлены в 19 и 20).
Анализ очередного значения происходит в перегруженном операторе 7. На вход подаются номер датчика и указатель на датчик. Если датчик работоспособный и его номер попадает в заданный диапазон номеров (строка 8), то в зависимости от параметра поиска через указатель вызывается соответствующая функция для анализа (строка 9), а также увеличивается счетчик просмотренных датчиков (строка 10). Функции 11 и 12 возвращают итоговые результаты.
Разработчику часто требуется много сторонних инструментов, чтобы создавать и поддерживать проект. Система Git — один из таких инструментов и используется для контроля промежуточных версий вашего приложения, позволяя вам исправлять ошибки, откатывать к старой версии, разрабатывать проект в команде и сливать его потом. В книге вы узнаете об основах работы с Git: установка, ключевые команды, gitHub и многое другое.В книге рассматриваются следующие темы:основы Git;ветвление в Git;Git на сервере;распределённый Git;GitHub;инструменты Git;настройка Git;Git и другие системы контроля версий.
Рассмотрено все необходимое для разработки, компиляции, отладки и запуска приложений Java. Изложены практические приемы использования как традиционных, так и новейших конструкций объектно-ориентированного языка Java, графической библиотеки классов Swing, расширенной библиотеки Java 2D, работа со звуком, печать, способы русификации программ. Приведено полное описание нововведений Java SE 7: двоичная запись чисел, строковые варианты разветвлений, "ромбовидный оператор", NIO2, новые средства многопоточности и др.
В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.
В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.
В книге рассказывается история главного героя, который сталкивается с различными проблемами и препятствиями на протяжении всего своего путешествия. По пути он встречает множество второстепенных персонажей, которые играют важные роли в истории. Благодаря опыту главного героя книга исследует такие темы, как любовь, потеря, надежда и стойкость. По мере того, как главный герой преодолевает свои трудности, он усваивает ценные уроки жизни и растет как личность.
Python - объектно-ориентированный язык сверхвысокого уровня. Python, в отличии от Java, не требует исключительно объектной ориентированности, но классы в Python так просто изучить и так удобно использовать, что даже новые и неискушенные пользователи быстро переходят на ОО-подход.