仿函数(functor),就是使一个类或类模板的使用看上去象一个函数。其实现就是类或类模板中对operator()进行重载,这个类或类模板就有了类似函数的行为。仿函数是智能型函数就好比智能指针的行为像指针,其就可看作是一个指针。但是智能指针是定义的一个类对象,所以在具备指针功能的同时也有其他的能力。仿函数的能力也可以超越operator()。因为仿函数可以拥有成员函数和成员变量,这意味着仿函数拥有状态。另一个好处是可以在执行期初始化它们。

预定义的仿函数

算术类

加法:plus<T>;减法:minus<T>;乘法:multiplies<T>;除法:divides<T>;求余:modulus<T>;否定:negate<T>

关系运算类

等于:equal_to<T>;不等于:not_equal_to<T>;大于:greater<T>;大于等于:greater_equal<T>;小于:less<T>;小于等于:less_equal<T>

逻辑运算类

除了否定为一元,其他都为二元仿函数。与:logical_and<T>;或:logical_or<T>;否:logical_not<T>

  下面是一个哈夫曼编码示例,遇到了需要对一个文件中的所有字符进行权重计算以创建每个字符的最终编码的过程,这其中有一个步骤是需要遍历已有的字符权重表以查找当前从文件中取得的字符是否已经存在于该表中,如果存在就将该表中的该字符权重加一,如不存在则需要新建一个存储 node 以存储该字符,并将该 node 结构添加到已有的权重表中。考虑到需要在权重表中进行字符查找以及之后创建 Huffman Tree 时可能需要对该表各项进行排序所以选用 vector<Node>作为存储容器,查找及排序可以使用 algorithm 头文件当中的 sort 和 find 算法。

 #include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include <algorithm>
#include <functional>//for bind2nd func; #ifndef HUFFMAN_H
#define HUFFMAN_H using namespace std; class Huffman
{
public:
Huffman(void);
//huffman(File file, File srcTable);
void encode(const string& file);
void decode(const string& file, const string& srcTable) const;
~Huffman();
private:
//table node
typedef struct
{
char letter;
int level;
int parent;
int direction;//-1 for no parent, 0 for left, 1 right;
}Node;
vector<Node> nodeTable;//can be sorted; //仿函数,用于find_if,侦测字符是否已经存在;
//template <typename T1, typename T2>
class Comparer : public binary_function<Node, char, bool>
{
public:
Comparer(const char ch):_ch(ch){}; const bool operator()(const vector<Node>::value_type& node, char ch) const
{
return node.letter == ch;
} private:
char _ch;
};
}; #endif
 #include "Huffman.h"

 Huffman::Huffman(void)
{
//dummany;
} void Huffman::encode(const string& file)
{
ifstream fin;
fin.open(file.c_str());
char ch;
while(fin.get(ch))
{
if (nodeTable.size() != )
{
//仿函数
vector<Node>::iterator result = find_if(nodeTable.begin(),nodeTable.end(),bind2nd(Comparer(ch), ch)); if (result == nodeTable.end())
{
Node* node = new Node;
node->letter = ch;
node->level = ;
node->parent = ;
node->direction = -;
}
else
{
result->level += ;
}
}
else
{
Node* node = new Node;
node->letter = ch;
node->level = ;
node->parent = ;
node->direction = -;
}
}
fin.close(); //huffman tree; } void Huffman::decode(const string& file, const string& srcTable) const
{
//dummany;
} Huffman::~Huffman(void)
{
//dummany;
}

  此处仿函数调用过程是这样的:find_if(nodeTable.begin(), nodeTable.end(), Comparer(ch))中 Comparer(ch)只是创建了 Comparer 类的匿名方法,重载的 operator() 真正的调用是在接下来将要看到的 find_if 模板函数的这一句 pred(*first);代码中不使用 find 而使用 find_if 是因为需要进行查找的不是 prime type 而是自行编写的符合类型,find_if 的函数原型参考如下,从原型中可以看到第一个参数是以默认的方式进行的:

 template<class InputIterator, class Predicate>
InputIterator find_if ( InputIterator first, InputIterator last, Predicate pred )
{
for ( ; first!=last ; first++ ) if ( pred(*first) ) break;
return first;
}

  bind2nd 函数的作用是将 二元算子(binary functor, bf) 转化为一元算子(unary functor, uf) 还有一个类似的 bind1st ,使用时需要包含 functional 头文件;进行比较的仿函数需要继承 binary_functor<typename T1,typename T2,typename T3>,T3 一般为 bool 值, binary_functor 原型如下:

 template<class Arg1,class Arg2,class Result>
struct binary_function
{
typedef Arg1 first_argument_type;
typedef Arg2 second_argument_type;
typedef Result result_type;
};

  这里使用的是 bind2nd 直接将得到的 char 参数 bind 到 Comparer 仿函数中,当然可以使用直接传参数到 Comparer 仿函数中人的方法,从 Huffman.h 和 Huffman.cpp 的注释中可以看到存在不同的可行方法。

最新文章

  1. SQL/LINQ/Lamda
  2. 安装Arch Linux
  3. 5天玩转C#并行和多线程编程
  4. Hive UDF’S addMonths
  5. VSS 访问问题
  6. Android so lib库远程http下载和动态注册
  7. hdu2647 拓扑序
  8. (XAML)&quot;XXXX&quot; does not exist in the namespace &quot;clr-
  9. php正则表达式获取表格内容
  10. delphi 反射(原理)
  11. BestCoder 百度之星2016
  12. 推荐一个优秀的c++源代码,TinyXml2
  13. 小甲鱼零基础python课后题 P24 023递归:这帮小兔崽子
  14. 关于OSI
  15. Maven问题:Failure to transfer org.apache.maven
  16. 湖南大学第十四届ACM程序设计新生杯(重现赛)
  17. windows程序设计 创建一个新的窗口
  18. UVA 10870 Recurrences(矩阵乘法)
  19. 算法笔记--单调队列优化dp
  20. python hashable

热门文章

  1. [Unity菜鸟] 射线
  2. Arraysort
  3. Android:Context的作用
  4. 在浏览器控制台输出内容 console.log(string);
  5. button 事件属性
  6. Android Include标签
  7. Servlet概述
  8. IOS地址
  9. kafka的环境搭建
  10. BZOJ 3123 森林(函数式线段树)