Додаткова функціональність в алгоритмах

Содержание

Слайд 2

Розширення STL функціями користувача

Передача функції через аргумент – стандартний механізм розширення

Розширення STL функціями користувача Передача функції через аргумент – стандартний механізм розширення
функціональності бібліотеки
Надійність механізму
неформальний опис стандартних вимог до прототипу функції
строга перевірка при компілюванні правильності оголошення функції
Гнучкість механізму
Може буди зовнішньою функцією або перевантаженим оператором виклику функції

template inline
Funct for_each(In first, In last, Funct f){
// виконання функції для кожного елемента
for (; first != last; ++first)
f(*first);
return f;
}

Слайд 3

Предикати

Окрема категорія функцій, які передають у алгоритми для перевірки аргумента
повертає bool
результат залежить

Предикати Окрема категорія функцій, які передають у алгоритми для перевірки аргумента повертає
лише від аргументів, не залежить від послідовності виклику
може буди зовнішньою функцією або перевантаженим оператором виклику функції (константним)
Унарні та бінарні предикати

template inline
_InIt _Find_if(_InIt _First, _InIt _Last, _Pr _Pred){
for (; _First != _Last; ++_First)
if (_Pred(*_First)) break;
return (_First);
}

Слайд 4

Об'єкти-функції

Об'єкти-функції – абстрактна категорія С++:
все, що веде себе як функція, є функцією
Крім

Об'єкти-функції Об'єкти-функції – абстрактна категорія С++: все, що веде себе як функція,
виклику звичайних функцій – об'єкт типу, в якому перевантажено operator()(...), у відповідному виразі зі списком аргументів для цього оператора

Слайд 5

Переваги об'єктів-функцій

Можливості типу не обмежуються наявністю operator()(...), це може бути повноцінний тип

Переваги об'єктів-функцій Можливості типу не обмежуються наявністю operator()(...), це може бути повноцінний
С++ з додатковими даними та функціональністю
Придатність до використання у ролі назви типу в аргументах функцій та шаблонів
При передачі об'єктів-функцій вирази можуть бути оптимізованими, більшість операторів перевантажують як inline

Слайд 6

Accumulate (сума елементів послідовності)

template
T accumulate (In first, In last,

Accumulate (сума елементів послідовності) template T accumulate (In first, In last, T
T init)
{
while (first!=last) {
init = init + *first;
++first;
}
return init;
}

int sum = accumulate(v.begin(),v.end(),0); // sum = 10

Слайд 7

Accumulate (сума елементів послідовності)

void f(vector& vd, int* p, int n)
{
double sum =

Accumulate (сума елементів послідовності) void f(vector & vd, int* p, int n)
accumulate(vd.begin(), vd.end(), 0.0);
// тип 3-о аргументу, ініціалізатора, визначає точність
int si = accumulate(p, p+n, 0); // небезпека переповнення
long sl = accumulate(p, p+n, long(0)); // результат в long
double s2 = accumulate(p, p+n, 0.0); // результат в double
// визначаєм змінну для результату і ініціалізації:
double ss = 0; //обов”язкова ініціалізація
ss = accumulate(vd.begin(), vd.end(), ss);
}

Слайд 8

Accumulate (узагальнення: обробка елементів послідовності)

Цим алгоритмом можна не тільки сумувати елементи послідовності, а

Accumulate (узагальнення: обробка елементів послідовності) Цим алгоритмом можна не тільки сумувати елементи
здійснювати над ними довільну бінарну операцію (н.п., *)
будь-яка функція, яка “оновлює величину init ” може бути використана:
template
T accumulate(In first, In last, T init, BinOp op)
{
while (first!=last) {
init = op(init, *first); // означає init op *first
++first;
}
return init;
}

Слайд 9

Accumulate

// перемноження елементів послідовності:
#include
void f(list& ld)
{
double product = accumulate(ld.begin(), ld.end(), 1.0,

Accumulate // перемноження елементів послідовності: #include void f(list & ld) { double
multiplies());
// …
}
multiplies – стандартний бібліотечний функціональний об”єкт для множення

множення *

ініціалізація 1.0

template
struct multiplies : public binary_function
{
Type operator()( const Type& _Left, const Type& _Right ) const;
};

Слайд 10

Accumulate (дані є частиною об’єкта)

struct Tovar {
int units;
double unit_price;
// …
};

Для ініціалізації та

Accumulate (дані є частиною об’єкта) struct Tovar { int units; double unit_price;
зберігання результату

double price (double v, const Tovar& r)
{
return v + r.unit_price * r.units;
}

void f(const vector& vr)
{
double total = accumulate(vr.begin(), vr.end(), 0.0, price);
// …
}

Слайд 11

class NewPrice
{
double zleva;
public:
NewPrice(double v):zleva(v){};
double operator()(double v,const Tovar& r){
return v + r.unit_price *

class NewPrice { double zleva; public: NewPrice(double v):zleva(v){}; double operator()(double v,const Tovar&
r.units*zleva;}
};

Accumulate (добуток елементів на певне значення)

void f(const list& vr)
{
double total = accumulate(vr.begin(), vr.end(), 0.0, NewPrice(0.1));
// …
}

Слайд 12

Використання предикатів

//предикат – функція
// переміщення в b елементів не більших за 15
bool

Використання предикатів //предикат – функція // переміщення в b елементів не більших
gt15(int x) { return 15 < x; };
int main() {
int a[]={10,20,30};
vector b;
remove_copy_if(a,a+3,back_inserter(b), gt15); //b has 10 only
...
}

Слайд 13

// предикат – функціональний об”єкт
// переміщення в b елементів не більших за

// предикат – функціональний об”єкт // переміщення в b елементів не більших
val
class gt_n {
int value;
public:
gt_n(int val) : value(val) {}
bool operator()(int n) { return n > value; }
};
int main() {
int a[]={1, 2, 3, 4, 5, 6, 7, 8,9 };
vector b;
gt_n f(4);
remove_copy_if(a, a+3, back_inserter(b), f);
remove_copy_if(a, a+3, back_inserter(b), gt_n(6));

Використання предикатів

Слайд 14

стандартні об'єкти-функції


Класифікація функціональних об”єктів бібліотеки STL здійснюється:
за кількістю аргументів їх

стандартні об'єкти-функції Класифікація функціональних об”єктів бібліотеки STL здійснюється: за кількістю аргументів їх
оператора operator( )
за типом значення, що повертається
Генератор (Generator) - тип функціонального об”єкта, що не має аргументів і повертає величину довільного типу
Приклад: функція rand( ) (визначений в ), застосовний до алгоритму generate_n( )
- Унарна функція – має один аргумент деякого типу і повертає величину різного типу (може бути void).
- Бінарна функція – має два аргументи деяких типів і повертає величину різного типу (може бути void).
- Унарний предикат – унарна функція, що повертає bool.
- Бінарний предикат – бінарна функція, що повертає bool.

Слайд 15

Шаблони базових типів стандартних об'єктів-функцій

template
struct unary_function
{// base class for unary

Шаблони базових типів стандартних об'єктів-функцій template struct unary_function {// base class for
functions
typedef _Arg argument_type;
typedef _Result result_type;
};
template
struct binary_function
{// base class for binary functions
typedef _Arg1 first_argument_type;
typedef _Arg2 second_argument_type;
typedef _Result result_type;
};

Слайд 16

Всі визначені в бібліотеці функціональні об’єкти є шаблонними типами і представляють всі

Всі визначені в бібліотеці функціональні об’єкти є шаблонними типами і представляють всі
вбудовані знаки арифметичних операцій, операції порівняння і логічні операції.
Параметрами шаблонів є типи елементів, над якими виконуються операції, які при використанні в алгоритмах повинні співпадати з типами елементів у контейнерах.
Параметри шаблону - довільні вбудовані типи С++ або типи, створені користувачем, для яких перевантажені відповідні оператори.

Стандартні об'єкти-функції

Слайд 17

Стандартні об'єкти-функції

Стандартні об'єкти-функції

Слайд 18

negate

template
struct negate
: public unary_function<_Ty, _Ty>
{// functor for unary operator-
_Ty operator()(const

negate template struct negate : public unary_function {// functor for unary operator-
_Ty& _Left)const
{// apply operator- to operand
return (-_Left);
}
};

Слайд 19

plus

template
struct plus
:public binary_function<_Ty,_Ty,_Ty>
{// functor for operator+
_Ty operator()(const _Ty& _Left,
const

plus template struct plus :public binary_function {// functor for operator+ _Ty operator()(const
_Ty& _Right) const
{// apply operator+ to operands
return (_Left + _Right);
}
};

Слайд 20

logical_or

template
struct logical_or
:public binary_function<_Ty,_Ty, bool>
{// functor for operator||
bool operator()(const _Ty&

logical_or template struct logical_or :public binary_function {// functor for operator|| bool operator()(const
_Left,
const _Ty& _Right) const
{// apply operator|| to operands
return (_Left || _Right);
}
};

Слайд 21

Використання стандартних об”єктів-функцій

// копіює елементи послідовності [first, last) у result, попередньо застосувавши

Використання стандартних об”єктів-функцій // копіює елементи послідовності [first, last) у result, попередньо
до кожного елемента функціональний об’єкт op:
template
Out transform(In first, In last, Out result, UnaryOper op)
// застосовує функціональний об’єкт op до кожної пари з вказаних діапазонів.
template
Out transform(In1 first1,In1 last1,In2 first2,Out result, BinOper op)

vector v,v1,v2;
// всі елементи з v запишуться в out з протилежним знаком
transform(v.begin(), v.end(), out, negate());
// додавання двох векторів поелементно
transform(v1.begin(), v1.end(), v2.begin(), out, plus());

Слайд 22

Vector sklad1, sklad2;
// зaповнення товаром sklad1, sklad2;
transform(sklad1.begin(), sklad1.end(), sklad2.begin(), sklad1, minus());
//

Vector sklad1, sklad2; // зaповнення товаром sklad1, sklad2; transform(sklad1.begin(), sklad1.end(), sklad2.begin(), sklad1,
для класу Tovar повинен бути перевантажений operator-
// Tovar& operator-(Tovar& t1, Tovar & t2)

Слайд 23

Адаптери об'єктів-функцій

Як використати бінарний функціональний об”єкт divides для ділення вектора на число?

Адаптери об'єктів-функцій Як використати бінарний функціональний об”єкт divides для ділення вектора на
Як використати предикат для оберненої операції?
Адаптери конвертують функціональний об”єкт до вимог алгоритмів, змінюючи кількість аргументів чи тип, що повертається
Функціональний об”єкт може бути адаптований завдяки заданню типу аргументів і значення, що повертається через typedef (argument_type і result_type).
Визначені в заголовковому файлі

Слайд 24

Адаптери стандартних об'єктів-функції

Адаптери стандартних об'єктів-функції

Слайд 25

Адаптери binder1st і binder2nd

Класи-адптери binder1st і binder2nd, конвертують адаптовані бінарні функції в

Адаптери binder1st і binder2nd Класи-адптери binder1st і binder2nd, конвертують адаптовані бінарні функції
адаптовані унарні функції, фіксуючи та зберігаючи перший (другий) аргумент як поле адаптера.
Функції-адаптери binder1st() і bind2nd() повертають об”єкти однойменних класів-адаптерів.

Слайд 26

class binder2nd

template// functor adapter _Func(left, stored)
class binder2nd
: public unary_function typename

class binder2nd template // functor adapter _Func(left, stored) class binder2nd : public
_Fn2::result_type>{
public:
typedef unary_function typename _Fn2::result_type> _Base;
typedef typename _Base::argument_type argument_type;
typedef typename _Base::result_type result_type;
. . .
protected:
_Fn2 op; // the functor to apply
typename _Fn2::second_argument_type value; // the right operand
};

Слайд 27

class binder2nd-2

template
class binder2nd : public unary_function ....
binder2nd(const _Fn2& _Func, const typename

class binder2nd-2 template class binder2nd : public unary_function .... binder2nd(const _Fn2& _Func,
_Fn2::second_argument_type& _Right)
: op(_Func), value(_Right) {}
result_type operator()(const argument_type& _Left) const {
return (op(_Left, value)); }
result_type operator()(argument_type& _Left) const{
return (op(_Left, value));}
};

Слайд 28

bind2nd()

template inline
binder2nd bind2nd(const Fn2& func,
const T& right)
{// return

bind2nd() template inline binder2nd bind2nd(const Fn2& func, const T& right) {// return
a binder2nd functor adapter
typename Fn2::second_argument_type
val(right);
return binder2nd(func, val);
}

Слайд 29

class binder1st

template
class binder1st: public unary_function
{ // functor adapter _Func(stored,

class binder1st template class binder1st: public unary_function { // functor adapter _Func(stored,
right)
public:
typedef unary_function typename _Fn2::result_type> _Base;
typedef typename _Base::argument_type argument_type;
typedef typename _Base::result_type result_type;
. . . . .
protected:
_Fn2 op; // the functor to apply
typename _Fn2::first_argument_type value;// the left operand
};

Слайд 30

class binder1st-2

template
class binder1st : public unary_function
. . .
binder1st(const _Fn2&

class binder1st-2 template class binder1st : public unary_function . . . binder1st(const
_Func,
const typename _Fn2::first_argument_type& _Left)
: op(_Func), value(_Left) {}
result_type operator()(const argument_type& _Right)
const {return (op(value, _Right));}
result_type operator()(argument_type& _Right) const
{return (op(value, _Right));}
};

Слайд 31

Використання адаптерів

vector v1(4,4);
ostream_iterator out(cout," ");
transform(v1.begin(), v1.end(), out,
bind2nd(divides (), 2));
// 2

Використання адаптерів vector v1(4,4); ostream_iterator out(cout," "); transform(v1.begin(), v1.end(), out, bind2nd(divides (),
2 2 2
transform(v1.begin(), v1.end(), out,
bind1st(divides (), 2));
// ?

Слайд 32

Крім бібліотечних функціональних об”єктів можна конвертувати з допомогою алаптерів й функціональні об’єкти,

Крім бібліотечних функціональних об”єктів можна конвертувати з допомогою алаптерів й функціональні об’єкти,
створені користувачем.
Для цього утворюють класи функціональних об’єктів, похідні від класів unary_function чи binary_function.
В якості параметрів шаблонів передають типи, що будуть типами аргументів і типом результату

class mult: public binary_function
{
public:
double operator()(const double &x, const double &y) const {return x* y;}
};
void main(){
vector v1(4,4);
ostream_iterator out(cout," ");
transform(v1.begin(), v1.end(), out, bind2nd(mult(), 2));

Слайд 33

class unary_negate

//Шаблонний клас забезпечує функцію, яка заперечує значення, яке повертає унарна функція
template

class unary_negate //Шаблонний клас забезпечує функцію, яка заперечує значення, яке повертає унарна
Fn1>
class unary_negate
: public unary_function
{
public:
explicit unary_negate(const Fn1& func):
functor( func) {}
bool operator()(const typename
Fn1::argument_type& left)const
{return ( !functor( left)); }
protected:
Fn1 functor; // the functor to apply
};

Слайд 34

not1()

template inline
unary_negate not1(const Fn1& func)
{ // return a unary_negate functor adapter
return std::unary_negate

not1() template inline unary_negate not1(const Fn1& func) { // return a unary_negate
(func);
}

Слайд 35

class binary_negate

template
class binary_negate: public binary_function
typename Fn2::second_argument_type, bool> {
public:
explicit

class binary_negate template class binary_negate: public binary_function typename Fn2::second_argument_type, bool> { public:
binary_negate(const Fn2& func):
functor(func){}
bool operator()(
const typename Fn2::first_argument_type& left,
const typename Fn2::second_argument_type& right
) const {
return (!functor(left, right));
}
protected:
Fn2 functor;
};

Слайд 36

not2()

template inline
binary_negate not2(const Fn2& func)
{
return std::binary_negate(func);
}

not2() template inline binary_negate not2(const Fn2& func) { return std::binary_negate (func); }

Слайд 37

class pointer_to_unary_function

//адаптер отримує вказівник на функцію і перетворює її (повертає) у функціональний

class pointer_to_unary_function //адаптер отримує вказівник на функцію і перетворює її (повертає) у
об’єкт. Працює лише для унарних та бінарних функцій (не для функцій без аргументів)
template class Fn = Result (*)(Arg)>
class pointer_to_unary_function
: public unary_function
{
public:
explicit pointer_to_unary_function(Fn left)
: pfun(left) {}
Result operator()(Arg left) const
{ // call function with operand
return (pfun(left));
}
protected:
Fn pfun; // the function pointer
};

Слайд 38

ptr_fun()

template inline
pointer_to_unary_function Result (__cdecl *)(Arg)>
ptr_fun( Result(__cdecl *

ptr_fun() template inline pointer_to_unary_function Result (__cdecl *)(Arg)> ptr_fun( Result(__cdecl * left)(Arg) ){
left)(Arg) ){
// return pointer_to_unary_function functor adapter
return
std::pointer_to_unary_function

(left);
}

Слайд 39

ptr_fun()-2

template< class Arg1, class Arg2,
class Result> inline
pointer_to_binary_function< Arg1, Arg2, Result,

ptr_fun()-2 template class Result> inline pointer_to_binary_function ptr_fun ( Result(__cdecl * left)(Arg1, Arg2)
Result (__cdecl *)(Arg1, Arg2)>
ptr_fun
( Result(__cdecl * left)(Arg1, Arg2) ){
// return pointer_to_binary_function functor adapter
return
pointer_to_binary_function
Result(__cdecl*) (Arg1, Arg2)> (left);
}

Слайд 40

int d[ ] = { 123, 94, 10, 314, 315 };
bool isEven(int

int d[ ] = { 123, 94, 10, 314, 315 }; bool
x) { return x % 2 = = 0; }
int main() {
  vector vb;
  transform(d, d + 5, back_inserter(vb),
not1(ptr_fun(isEven)));
  // vb: 1 0 0 0 1
}

Слайд 41

double d[] = { 01.23, 91.370, 56.661,023.230, 19.959, 1.0, 3.14159 };
int main()

double d[] = { 01.23, 91.370, 56.661,023.230, 19.959, 1.0, 3.14159 }; int
{
vector vd;
transform(d, d + 7,back_inserter(vd),
bind2nd(ptr_fun(pow), 2.0));
copy(vd.begin(),vd.end(),ostream_iterator(cout, " "));
}

Слайд 42

Адаптери mem_fun()та mem_fun_ref_t

Адаптер mem_fun( ) утворює функціональний об”єкт, який викликається, використовуючи вказівник на

Адаптери mem_fun()та mem_fun_ref_t Адаптер mem_fun( ) утворює функціональний об”єкт, який викликається, використовуючи
об”єкт, для якого треба викликати функцію-член (метод)
Адаптер mem_fun_ref( ) викликає метод прямо для об”єкта.

Слайд 43

class mem_fun_t

template
class mem_fun_t
: public unary_function< T*, Result> {
//

class mem_fun_t template class mem_fun_t : public unary_function { // functor adapter
functor adapter (*p->*pfunc)(), non-const
public:
explicit mem_fun_t(Result (T::* pm)())
: pmemfun(pm){}
Result operator()(T * pleft) const
{ // call function
return ((pleft->*pmemfun)());
}
private:
Result(T::*pmemfun)();// the member function pntr
};

Слайд 44

mem_fun()

template inline
mem_fun_t
mem_fun(Result (T::* pm)())
{ // return a mem_fun_t

mem_fun() template inline mem_fun_t mem_fun(Result (T::* pm)()) { // return a mem_fun_t
functor adapter
return mem_fun_t(pm);
}

template inline
const_mem_fun_t
mem_fun(Result (T::* pm)() const)
{ // return a const_mem_fun_t functor adapter
return const_mem_fun_t(pm);
}

Слайд 45

class Shape {
public:
virtual void draw() = 0;
virtual ~Shape() {}
};
class Circle

class Shape { public: virtual void draw() = 0; virtual ~Shape() {}
: public Shape {
public:
virtual void draw() { cout << "Circle::Draw()" << endl; }
~Circle() { cout << "Circle::~Circle()" << endl; }
};
class Square : public Shape {
public:
virtual void draw() { cout << "Square::Draw()" << endl; }
~Square() { cout << "Square::~Square()" << endl; }
};
int main() {
vector vs;
vs.push_back(new Circle);
vs.push_back(new Square);
for_each(vs.begin(), vs.end(), mem_fun(&Shape::draw));
} ///:~

Слайд 46

class mem_fun_ref_t

template
class mem_fun_ref_t
: public unary_function {
// functor adapter

class mem_fun_ref_t template class mem_fun_ref_t : public unary_function { // functor adapter
(*left.*pfunc)(), non-const *pfunc
public:
explicit mem_fun_ref_t(Result (T::* pm)())
: pmemfun(pm){}
Result operator()(T& left) const {
return (left.*pmemfun)();
}
private:
Result(T::*memfun)();// the member function pointer
};

Слайд 47

mem_fun_ref()

template inline
mem_fun_ref_t
mem_fun_ref(Result (T::* pm)())
{ // return a

mem_fun_ref() template inline mem_fun_ref_t mem_fun_ref(Result (T::* pm)()) { // return a mem_fun_ref_t
mem_fun_ref_t functor adapter
return mem_fun_ref_t(pm);
}

Слайд 48

class Angle {
  int degrees;
public:
  Angle(int deg) : degrees(deg) {}
  int mul(int times)

class Angle { int degrees; public: Angle(int deg) : degrees(deg) {} int
{ return degrees *= times; }
};
int main() {
  vector va;
  for(int i = 0; i < 50; i += 10)
    va.push_back(Angle(i));
  int x[] = { 1, 2, 3, 4, 5 };
  transform(va.begin(), va.end(), x,
    ostream_iterator(cout, " "),
    mem_fun_ref(&Angle::mul));
  cout << endl;
  // Output: 0 20 60 120 200
} ///:~

Слайд 49

mem_fun1_t

template
class mem_fun1_t
: public binary_function

mem_fun1_t template class mem_fun1_t : public binary_function { // functor adapter (*p->*pfunc)(val),
{
// functor adapter (*p->*pfunc)(val), non-const *pfunc
public:
explicit mem_fun1_t(Result (T::* pm)(Arg))
: pmemfun(pm){}
Result operator()(T *pleft, Arg right) const
{ // call function with operand
return ( pleft->*pmemfun)(right);
}
private:
Result(T::* pmemfun)(Arg);//the member function pntr
};
Имя файла: Додаткова-функціональність-в-алгоритмах-.pptx
Количество просмотров: 108
Количество скачиваний: 0