Лекция 5. bind _ function (1114997)
Текст из файла
Лекция 5. bind & functionValery Lesin. C++ In-Depth, 20141Задача: вызов callback’a1.2.3.4.5.6.7./*(1)*/void set_timer(void(*)(double /*time*/));/*(2)*/void set_timer(void(*)(double /*time*/, void* /*data(1)*/));/*(3)*/template<class T> void set_timer(T* obj, void (T::*)(double));/*(4)*/template<class F> void set_timer(F const& f); // f(time)/*(5)*/void set_timer(ITimeHandler* handler);1. Не передать данные.2. Передать, но не typesafe. Нужно формироватьструктуру.3.
Функций set_timer по количеству клиентов.Нужно иметь правильную функцию в типе Т.4. Можно и без правильной функции, но придетсяделать подходящий функтор.5. Одна реализация, но требует полиморфныйинтерфейс от класса.Valery Lesin. C++ In-Depth, 20142Немного магии. boost::bind• bind обобщает каррирование для функцийнескольких аргументов.1.2.3.4.5.void foo(int a, string b, double c);bind(foo, 1, "2", 3.)();bind(foo, 1, _1 , 3 )("2");bind(foo, _2, _3, _2)(2, 1, "2", 7);Valery Lesin. C++ In-Depth, 2014// foo(1, "2", 3.)// foo(1, "2", 3.)// foo(1, "2", 1.)3Как избежать копирования? ref/cref• bind копирует переданные параметры.
Ноэтого можно избежать.1.2.3.4.5.6.7.8.void foo(vector<string> const& vstr, int value) {/*...*/}//..vector<string> v(1000);// makes copyauto a = bind(foo, v, _1);// no copy, used reference_wrapperauto b = bind(foo, cref(v), _1);Valery Lesin. C++ In-Depth, 20144Использование bind c функторами1.2.3.4.5.6.7.8.9.10.11.struct func_t{string operator()(intto_str) const{/*...*/}intoperator()(string to_int) const{/*...*/}};//..func_t f;bind<int>(f)("42"); // funct_t::operator()(string)bind<string>(f)(24);// funct_t::operator()(int)• Перегрузка по-прежнему выполняется попереданному параметру.• Если функтор предоставляет result_type (aля std::unary_function), то возвращаемыйтип можно не указывать.Valery Lesin.
C++ In-Depth, 20145Вызов методов1.2.3.4.5.6.7.8.9.10.11.12.13.14.struct T{void func(string str) {/*...*/}};T obj;shared_ptr<T> ptr(new T);string name = "Edgar";bind(&T::func,bind(&T::func,bind(&T::func,bind(&T::func,ref(obj),&obj,obj,ptr,_1)(name);_1)(name);_1)(name);_1)(name);• Вызовы 13-14 держат объект, пока существует самбиндер.• Бинд виртуальных функций работает корректно.Valery Lesin.
C++ In-Depth, 20146Вложенность bind• Как поступить, если параметр еще незафиксирован, но его генератор известен?1.2.3.4.5.void foo(int x, int y);int bar(int x);auto b = bind(foo, bind(bar, _1), _2);b(a, b); // foo(bar(a), b)• А можно ли не фиксировать саму функцию?1.2.3.4.int bar2(int x);auto bb = bind(apply<int>(), _1, 42);bb(bar2);Valery Lesin. C++ In-Depth, 20147boost::function• Контейнер функции с фиксированнымпрототипом:– есть функции empty и clear;– можно использовать в условных выражениях.1.2.3.4.5.void foo(double a, double b) {/*...*/}//..function<void(int x, double y)> f = foo;f(3, 14);Valery Lesin.
C++ In-Depth, 20148Синергический эффект bind & function1.2.3.4.5.6.7.8.9.typedef function<void(double/*time*/)> timer_f;// just one function for all clientsvoid set_timer(timer_f const& f);//..set_timer(bind(&window_t::redraw, &wnd_));set_timer(bind(&beeper_t::make_sound, &beeper_, 440));set_timer(bind(&clock_t ::update, &clock_ , _1));• Всего одна нешаблонная (!) реализация.• Допускаются дополнительные данные.• Не требуется определять дополнительныефункторы.• Type safe.• Нет необходимости в полиморфном интерфейсе ивиртуальных функциях (простор для оптимизатора).Valery Lesin. C++ In-Depth, 20149Быстродействие• Дополнительное время уходит на выделенияпамяти.
А, если их нет – на косвенные вызовы.• bind не делает дополнительных выделенийпамяти. Косвенный вызов функции можетбыть оптимизирован в bind тогда же, когда ибез него.• function может выделять память под«большой» объект binder. Поэтому лишний разне копируйте function.Valery Lesin.
C++ In-Depth, 201410Как работают placeholder’ы?• Возможный путь реализации:1.2.3.4.5.6.7.8.9.10.11.12.13.14.15.16.17.18.19.20.21.22.23.24.template<class arg_t, class arg1_t, class arg2_t, class arg3_t>arg_t const& take_arg(arg_t const& arg, arg1_t const&,arg2_t const&, arg3_t const&){return arg;}template<size_t N, class arg1_t, class arg2_t, class arg3_t>typename N_th<N, arg1_t, arg2_t, arg3_t>::type const&take_arg(placeholder<N> const& pl, arg1_t const& arg1,arg2_t const&arg2, arg3_t const& arg3){return pl(arg1, arg2, arg3);}template<class arg1_t, class arg2_t, class arg3_t>auto binder_t::operator ()(arg1_t const& arg1, arg2_t const& arg2,arg3_t const& arg3) -> R{// K - number of func_ parametersfunc_(take_arg(pos1_, arg1, arg2, arg3),... ,take_arg(posK_, arg1, arg2, arg3));}Valery Lesin.
C++ In-Depth, 201411Вопросы?Valery Lesin. C++ In-Depth, 201412.
Характеристики
Тип файла PDF
PDF-формат наиболее широко используется для просмотра любого типа файлов на любом устройстве. В него можно сохранить документ, таблицы, презентацию, текст, чертежи, вычисления, графики и всё остальное, что можно показать на экране любого устройства. Именно его лучше всего использовать для печати.
Например, если Вам нужно распечатать чертёж из автокада, Вы сохраните чертёж на флешку, но будет ли автокад в пункте печати? А если будет, то нужная версия с нужными библиотеками? Именно для этого и нужен формат PDF - в нём точно будет показано верно вне зависимости от того, в какой программе создали PDF-файл и есть ли нужная программа для его просмотра.