1、 仿函数

仿函数又名函数对象。具有函数性质的对象。就是传入一些參数。然后对參数进行某些运算,然后返回一个值。

为了可以使行为类似函数,须要在类别定义中必须自己定义function call 运算子operator()。

仿函数有例如以下几类:算术类仿函数(plus<T>、minus<T>)关系运算类仿函数(equal_to<T>、less<T>)逻辑运算类仿函数(logical_and<T>、logical_or<T>、logical_not<T>)证同、选择、投射。

1.1、可配接的关键

可配接指的是可以在类原有的基础上扩展成另外一个类。仿函数为此定义了两个类,分别代表一元仿函数和二元仿函数。

之后的不论什么仿函数,依据功能需求选择当中一个类继承。详细代码例如以下:

一元仿函数

template<class Arg, classResult>

struct unary_function

{

typedef Arg argument_type;

typedef Result result_type;

};

template<class T>

struct negate:publicunary_function<T,T>

{

T operator()(const T& x) const {return -x;}

};

二元仿函数

template<class Arg1,classArg2,class Result>

struct binary_function

{

typedef Arg1 first_argument_type;

typedef Arg2 second_argument_type;

typedef Result result_type;

};

template<class T>

struct plus:publicbinary_function<T,T,T>

{

T operator()(const T& x,const T& y) const {return x+y;}

};

2、配接器

配接器类似于转换器,它是一种设计模式,在原有的类型基础上扩展成为另外一个接口,使原本由于接口不兼容而不能合作的类型能够一起工作。

配接器分类例如以下:function adapter(改变仿函数接口)、container adapter(改变容器接口)、iterator adapter(改变迭代器接口)。

2.1     container adapter

最明显的样例就是stack和queue, 直接上代码,一看便知:

template<class T,classSequence=deque<T>>

class stack

{

protected:

Sequence c; //底层容器採用 deque

}

template<class T.classSequence=deque<T>>

class queue

{

protected:

Sequence c;

}

由上可知,class stack和class queue均以deque为底层容器。然后通过封主deque全部对外接口,仅仅开放符合stack或queue原则的几个函数,所以我们称stack和queue是个配接器,一个作用于容器之上的配接器。

2.2     iterator adapter

书中针对迭代器适配器分三类进行描写叙述:insert iterator、 reverseiterator、iostream iterator。后两种在实现技巧和理解上都相对复杂一些。通过实例来分析吧,代码例如以下:

int main()

{

1    ostream_iterator<int> outite(cout," ");

2    int ia[]={0,1,2,3,4,5};

3    deque<int> id(ia,ia+6);

4    copy(id.begin(),id.end(),outite);

5    cout<<endl;  // 0 1 23 4 5

6    copy(ia+1,ia+2,front_inserter(id));

7    copy(id.begin(),id.end(),outite);

8    cout<<endl;  //1 0 12 3 4 5

9    deque<int>::iterator ite=find(id.begin(),id.end(),4);

10  reverse_iterator<deque<int>::iterator>rite(ite)

11   cout<<*ite<<endl; //4

12  cout<<*rite<<endl; //3

}

由上可知。上述第1行代码。採用了iostream iteratoradapter将迭代起绑定到cout对象,组成一个ostream_iterator。拥有输出功能。

第6行代码中front_inserter(id)是个辅助函数,方便client使用insert_iterator。调用该函数后实际产生一个对象迭代器适配器对象:

front_inserter_iterator<container>(x)。第10行代码中,通过reverse iterator迭代器配接器,转换了迭代器的方向,可是须要注意的是迭代器被逆转,尽管实体位置不变。即指针所在的位置不变。可是其所指的实体位置发生的改变,一个指向右,一个指向左。例如以下图所看到的:

2.3     function adapter

该部分是全部配接器中数量最庞大的一个族群。主要是通过对仿函数进行配接操作。组成更加丰富的表达式,简单的样例例如以下:

not1(bind2nd(less<int>(),12))

上式表述的是一个不小于12的表达式,当中less<int>()为仿函数,通过bind2nd这个配接器器辅助函数的操作转换成第二种仿函数。

另外须要补充的是用户函数指针的ptr_fun配接器和用户成员函数指针的mem_fun\men_fun_ref。通过配接器操作可以将一般函数和成员函数当做仿函数使用,并传给STL算法。

 

《STL源代码剖析》学习笔记系列之七、八——仿函数和配接器的更多相关文章

  1. python学习笔记系列----(八)python常用的标准库

    终于学到了python手册的最后一部分:常用标准库.这部分内容主要就是介绍了一些基础的常用的基础库,可以大概了解下,在以后真正使用的时候也能想起来再拿出来用. 8.1 操作系统接口模块:OS OS模块 ...

  2. Android学习笔记(十八)——使用意图筛选器和实现浏览网页(附源代码)

    使用意图筛选器 点击下载源代码 1.创建一个Intents项目,给该项目加入一个新类,命名为MyBrowserActivity.在res/layout目录下新增一个browser.xml: 2.在An ...

  3. Python--网络编程学习笔记系列01 附实战:udp聊天器

    Python--网络编程学习系列笔记01 网络编程基本目标: 不同的电脑上的软件能够实现数据传输 网络编程基础知识: IP地址: 用来在网络中标记一台电脑  网络号+主机号(按网络号和主机号占位分类A ...

  4. 《STL源代码剖析》---stl_deque.h阅读笔记(2)

    看完,<STL源代码剖析>---stl_deque.h阅读笔记(1)后.再看代码: G++ 2.91.57,cygnus\cygwin-b20\include\g++\stl_deque. ...

  5. .NET CORE学习笔记系列(2)——依赖注入[4]: 创建一个简易版的DI框架[上篇]

    原文https://www.cnblogs.com/artech/p/net-core-di-04.html 本系列文章旨在剖析.NET Core的依赖注入框架的实现原理,到目前为止我们通过三篇文章从 ...

  6. Dynamic CRM 2013学习笔记 系列汇总

    这里列出所有 Dynamic CRM 2013学习笔记 系列文章,方便大家查阅.有任何建议.意见.需要,欢迎大家提交评论一起讨论. 本文原文地址: Dynamic CRM 2013学习笔记 系列汇总 ...

  7. 【Unity Shaders】学习笔记——SurfaceShader(八)生成立方图

    [Unity Shaders]学习笔记——SurfaceShader(八)生成立方图 转载请注明出处:http://www.cnblogs.com/-867259206/p/5630261.html ...

  8. 步步为营 SharePoint 开发学习笔记系列总结

    转:http://www.cnblogs.com/springyangwc/archive/2011/08/03/2126763.html 概要 为时20多天的sharepoint开发学习笔记系列终于 ...

  9. STL源代码剖析 读书总结

    <<STL源代码剖析>> 侯捷著 非常早就买了这本书, 一直没看, 如今在实验室师兄代码的时候发现里面使用了大量泛型编程的内容, 让我有了先看看这本书的想法. 看之前我对于泛型 ...

随机推荐

  1. [CC-CHANOQ]Chef and odd queries

    题目大意: 给定$n(10^5)$个线段$[l_i,r_i](1\leq l_i,r_i\leq n)$,有$q(q\leq10^5)$组询问,每次给出$m_i(\sum m_i\leq n)$个点$ ...

  2. redis --- lua 脚本实现原子操作

    如题, 楼主的想法很简单, lua 脚本本身支持原子性, 所以把命令写进一个脚本就行, 当然后续还会优化才能放到生产上,例如缓存脚本 ,redis 本身会缓存执行过的脚本 ,这样速度更快, 再优化, ...

  3. redis 安装并且设置开机后台自动启动(转)

      1,安装redis wget http://download.redis.io/releases/redis-2.8.8.tar.gz .tar.gz cd redis- make 2,建立Red ...

  4. Bluetooth篇 开发实例之八 匹配

    自己写的App匹配蓝牙设备,不需要通过系统设置去连接. 匹配和通信是两回事. 用过Android系统设置(Setting)的人都知道蓝牙搜索之后可以建立配对和解除配对,但是这两项功能的函数没有在SDK ...

  5. 微服务实施Spring Boot/Spring Cloud中踩过的坑(转)

    http://tietang.wang/2016/09/08/%E5%BE%AE%E6%9C%8D%E5%8A%A1/%E5%BE%AE%E6%9C%8D%E5%8A%A1%E5%AE%9E%E6%9 ...

  6. ArcMAP定义投影坐标

    WGS84形式的高斯克吕格投影在ArcGIS系统中是不存在的,需要自己去定义.下面为MARK的定义过程.

  7. JS -- 一篇文章掌握RequireJS常用知识

    本文采取循序渐进的方式,从理论到实践,从RequireJS官方API文档中,总结出在使用RequireJS过程中最常用的一些用法,并对文档中不够清晰具体的内容,加以例证和分析,分享给大家供大家参考,具 ...

  8. Python命令行参数学习

    man python 查看python的帮助文件 命令行参数:       -B     Don't write .py[co] files on import.              See a ...

  9. spring boot 使用mybatis-generator

    mybatis-generator官网: http://www.mybatis.org/generator/running/runningWithMaven.html 在pom.xml中的 build ...

  10. mysql关于访问权限以及root密码修改

    root密码修改:mysql> use mysql;mysql> UPDATE user SET Password = PASSWORD('newpass') WHERE user = ' ...