`
美丽的小岛
  • 浏览: 309391 次
  • 性别: Icon_minigender_1
  • 来自: 大连
社区版块
存档分类
最新评论

STL函数对象及函数对象适配器【转】

    博客分类:
  • c++
  • c
 
阅读更多

 函数对象Functor

    STL中提供了一元和二元函数的两种Functor,通过unary_function和binary_function提供了这两种不同参数数量的Functor的基本结构,在这两个类型中,分别内嵌定义一元和二元函数操作在模版推演的时候需要用到的typedef.
 

//一元函数的定义为
template<class _A, class _R>
struct unary_function {
 typedef _A argument_type;
 typedef _R result_type;
};


//二元函数的定义为
template<class _A1, class _A2, class _R>
 struct binary_function {
 typedef _A1 first_argument_type;
 typedef _A2 second_argument_type;
 typedef _R result_type;
};

其他的一元和二元Functor可以从这两个基本结构继承,同时也就可以推演出函数的参数和返回值的类型,STL在上述这两个结构的基础上,实现了很多一元和二元的Functor.

//一元
negate

//二元
plus
minus
multiplies
divides
modulus
equal_to
not_equal_to
greater
greater_equal 
less
less_equal
logical_and
logical_or
logical_not

上面的这些Functor都是基于模版实现的,可以象下面那样使用的方式:

plus<int> int_plus;
cout << int_plus(111,222) << endl;


 函数对象适配器

函数对象适配器的作用就是使函数转化为函数对象,或是将多参数的函数对象转化为少参数的函数对象。


1)bind

bind1st  //通过绑定第一个参数,使二元的函数对象转化为一元的函数对象
bind2nd  //通过绑定第二个参数,使二元的函数对象转化为一元的函数对象
not1     //对一元的函数对象取反
not2     //对二元的函数对象取反

使用的方式:
bind1st( less<int>(), 10)(20);
not2( less<int() )(10,20);

这些Functor看起来好像好像用处不大,但是在和STL中的容器和算法结合在一起使用的时候,就会使得程序显得很简洁.

int i;    
vector<int> lv;
for(i = 0; i < 100; i++) 
{
    lv.push_back(i);
}
//对vector中小于20的数进行记数
cout << count_if(lv.begin(), lv.end(), bind2nd(less<int>(), 20)) << endl;

//由大到小排序
sort(lv.begin(), lv.end(), not2(less<int>()) ) ;
for (i = 0; i < 100; i++) 
{
    cout << lv.at(i) << endl;
}


2)ptr_fun

ptr_fun是指将现有的函数转换为Functor的功能.在STL中提供了这个功能的Functor,就是pointer_to_unary_function和pointer_to_binary_function这两个类,这两个类对应一元

和二元两种函数,也就是说,对于调用参数为3个或者多于3个的函数,STL提供的Functor类,无法配接.

基本使用方法:

int u_func(int a)
{
    int ret = a;
    return ret;
}   

int b_func(int a,int b)
{
    return a+b;
}

void call()
{
 pointer_to_unary_function<int,int> uf(u_func);
    cout << uf(100) << endl;
    
    pointer_to_binary_function<int,int,int> bf(b_func);
    cout << bf(111,222) << endl;

 //或者
 cout << ptr_fun(u_func)(100) << endl;
    cout << ptr_fun(b_func)(111,222) << endl;

}
可以看到,上面的方法改进了原先C和C++中通过函数指针来间接调用函数的方法,将函数指针封装到了类中.


问题:

第一部分中的Functor中是自己定义操作符(),但是在ptr_fun中,是将已经有的function转为Functor调用就会存在一个调用方式的问题.

c++中的函数,按调用方式可以分为__cdecl, __stdcall,__fastcall 三种,ptr_fun如何正确的识别给定的function的调用方式就会有问题.

其中:
vc6中的STL的ptr_fun代码中,统一将function认为是__cdecl调用方式. 而Dev-cpp中使用的SGI的代码中没有明确指明函数的调用方式,所以将使用编译器的确省设置.
但是如果将上面的b_func函数改为
int __stdcall b_func(int a,int b)
{
    return a+b;
}
上面的使用代码在DEV-CPP中无法编译通过.

3)mem_fun

mem_fun是将某个类中的成员函数转变为Functor的功能.

一般的使用方法
struct mem_fun_struct
{
    int n_mem_fun() {
        cout << "mem_fun_struct::n_mem_fun()" << endl;
        return 0;
    }    
    
    int u_mem_fun(int a) {
        cout << "mem_fun_struct::u_mem_fun(int) " << a << endl;
        return a;
    }    
    
    int b_mem_fun(int a,int b) {
        cout << "mem_fun_struct::b_mem_fun(int,int)" << a << " " << b << endl;
        return a+b;
    }    
};

void call()
{
 mem_fun_struct ls;
    mem_fun(&mem_fun_struct::n_mem_fun)(&ls);
    mem_fun(&mem_fun_struct::u_mem_fun)(&ls, 10);
    //mem_fun(&mem_fun_struct::u_mem_fun)(&ls, 10, 20);
}

上面的代码在dev-cpp 4.9.9中编译通过,SGI STL中没有提供二元成员函数的mem_fun,vc6中提供了mem_fun(无参数成员函数)和mem_fun1(一元参数成员函数), 而在vs2003中改变了用法.但是我看MSDN好像也只支持到一个参数.

 总结

STL中提供了基本的一元和二元参数的Functor, 同时提供了相应的适配器可以对Functor进行修饰,Functor可以很好的和 STL容器,STL算法结合使用.

但是仍有问题:
1)上面说到的调用方式
2) 多参数函数对象适配

对于我们比较复杂的stl不能满足要求的问题,我们可以是用boost或loki来解决。


 参考

本文基本是对hdqqq的文章转载,稍加整理!原文地址:http://www.cppblog.com/hdqqq/archive/2006/09/13/12424.aspx

同时参考msdn:http://msdn2.microsoft.com/en-us/library/4y7z5x4b(VS.80).aspx

 

Binder1st:
		// TEMPLATE CLASS binder1st
template<class _Fn2>
	class binder1st
		: public unary_function<typename _Fn2::second_argument_type,
			typename _Fn2::result_type>
	{	// functor adapter _Func(stored, right)
public:
	typedef unary_function<typename _Fn2::second_argument_type,
		typename _Fn2::result_type> _Base;
	typedef typename _Base::argument_type argument_type;
	typedef typename _Base::result_type result_type;

	binder1st(const _Fn2& _Func,
		const typename _Fn2::first_argument_type& _Left)
		: op(_Func), value(_Left)
		{	// construct from functor and left operand
		}

	result_type operator()(const argument_type& _Right) const
		{	// apply functor to operands
		return (op(value, _Right));
		}

	result_type operator()(argument_type& _Right) const
		{	// apply functor to operands
		return (op(value, _Right));
		}

protected:
	_Fn2 op;	// the functor to apply
	typename _Fn2::first_argument_type value;	// the left operand
	};


 

分享到:
评论

相关推荐

    每天学点C++(C++实例教程:教程+源码)STL函数对象.zip

    在C++编程语言中,STL(Standard Template Library,标准模板库)是其核心特性之一,它极大地丰富了C++的库功能,...通过学习和研究这些例子,你将能够熟练地利用STL函数对象来优化你的C++程序,提高代码质量和效率。

    C++进阶STL适配器总结1

    STL适配器则是为了增强或改变原有容器、迭代器或函数对象的行为。本文将对STL适配器进行总结,特别是关注其中的插入迭代器适配器。 首先,我们来看容器适配器。容器适配器是在原有容器的基础上进行封装,以实现特定...

    C++STL中文版 介绍了STL常用的函数和用法

    4. 仿函数(Functors)和函数对象:这是带有操作行为的对象,类似于C++中的函数指针。它们可以作为算法的参数,以自定义行为。比如,`std::less`和`std::greater`用于比较操作,`std::equal_to`用于判断相等。 5. ...

    函数对象.doc_cplus_plus

    C++标准库提供了诸如`std::bind`和函数对象适配器(如`std::mem_fun`、`std::ptr_fun`等),这些工具可以帮助将成员函数或非成员函数转换为函数对象,进一步增强函数对象的使用范围。 7. 对比函数指针: 尽管函数...

    复件 复件 函数对象--20100125.ppt

    此外,STL还提供了函数适配器(Function Adapters),如: 1. **绑定器(binder)**:`bind1st`和`bind2nd`用于固定一个或两个参数,将二元函数对象转化为一元函数对象。 2. **取反器(negator)**:`not1`和`not2`...

    C++常用stl算法.pdf

    这篇文档主要讨论了C++ STL中的一些常用算法,包括函数对象适配器,如`for_each`、`bind`等。让我们深入探讨这些概念。 首先,函数对象在C++中是一个重要的概念,它是一个类,通过重载`()`操作符使得类的对象能够像...

    c++STL容器适配器习题答案.docx

    C++的STL(Standard Template Library,标准模板库)是C++编程中极其重要的...STL的强大之处在于其灵活性和可组合性,可以根据不同的需求选择合适的容器、迭代器和算法,以及定制函数对象,以实现复杂的数据处理任务。

    STL源码剖析pdf及源码.zip

    STL是C++编程中不可或缺的一部分,它提供了高效、可重用的容器(如vector、list、set等)、迭代器(用于遍历容器)、算法(如排序、查找等)和函数对象(如比较函数、适配器等)。STL的实现基于泛型编程,利用模板...

    stl的介绍 STL算法作为模板函数提供

    1. **非面向对象设计**:STL的设计不依赖于面向对象的三大核心特性——封装、继承和多态。这种设计选择是为了提供更广泛的适用性和更高的性能。 2. **模板化**:STL中的容器、迭代器和算法都是以模板的形式提供的,...

    C++ STL教程pdf

    STL的核心概念包括泛型编程、容器、迭代器、算法和函数对象,这些组件共同构成了一个强大的抽象数据结构和算法库。 1. **泛型编程思想**:泛型编程是STL的基础,它允许编写不依赖于具体数据类型的代码。通过模板...

    Center of STL Study 标准模板库(STL)介绍

    **四、STL函数对象** 函数对象,也称为仿函数,是具有函数调用操作符的类对象。它们常用于算法中作为参数,实现特定的比较或操作。常见的函数对象有: 1. **比较函数对象**:如`less`、`greater`,用于排序和查找...

    STL Programmer's Guide(STL帮助文档)

    STL中的一些算法需要一个比较或操作函数,函数对象可以作为这些函数的参数,如less用于排序时的比较,加法运算符(+)可以被用作仿函数进行元素间的操作。 5. 配适器(Adapters): 配适器类可以改变现有容器或函数...

    STL源码剖析.简体中文 PDF

    STL中常见的函数对象包括比较函数对象(如`std::less`、`std::greater`)和操作函数对象(如`std::plus`、`std::multiplies`)。这些函数对象使得我们可以自定义算法的行为。 5. **适配器**:适配器允许将已有的...

    关于STL库中的函数的实现(标准版)

    STL的主要组件包括容器(如vector、list、set等)、迭代器、算法和函数对象。在这个标准版的讲解中,我们将深入探讨STL的关键知识点。 首先,我们来谈谈容器。容器是STL的核心,它们是用来存储和管理对象的类模板。...

    C++ STL程序员面试题

    - 函数对象(或称谓谓词):用于定义特定操作,如比较函数、函数对象适配器等。 2. **三十分钟掌握STL.doc** 这个文档可能旨在快速介绍STL的关键概念和常见用法。可能涵盖: - vector的高效插入和删除(在末尾)...

    C++STL源码剖析--侯捷

    5. **适配器**:适配器是STL中一个独特的设计,它允许将现有的容器、迭代器或函数对象进行包装,以适应新的需求。如stack和queue是基于现有容器的适配器,bind1st和bind2nd是函数对象适配器。 6. **内存管理**:STL...

    [中文]STL参考手册.zip

    适配器是STL中的一种设计模式,它可以使现有的容器、迭代器或函数对象具有新的行为。例如,stack和queue是容器适配器,它们在底层使用vector或deque,但提供了栈或队列的行为;`reverse_iterator`是迭代器适配器,它...

    STL源码剖析.pdf

    STL是C++标准库的一部分,提供了高效的数据结构和算法,如容器、迭代器、算法、函数对象和适配器等,广泛应用于现代C++编程中。 ### STL概述 STL的核心概念包括容器、迭代器、算法、函数对象和适配器。其中: - *...

    SGI_STL源码(附带个人详细)

    6. 适配器函数:如`bind1st`和`bind2nd`,以及`mem_fun`系列,它们用于封装成员函数,使非成员函数可以调用对象的成员方法。 在SGI_STL源码中,你可以看到每个组件的具体实现,例如: - `__alloc`目录包含内存分配...

Global site tag (gtag.js) - Google Analytics