C++函数对象与函数指针不同之处

2024-09-07 18:48
文章标签 c++ 函数 对象 函数指针

本文主要是介绍C++函数对象与函数指针不同之处,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在C++编程语言中,有很多功能都与C语言相通,比如指针的应用等等。在这里我们介绍的则是一种类似于函数指针的C++函数对象的相关介绍。C++函数对象不是函数指针。但是,在程序代码中,它的调用方式与函数指针一样,后面加个括号就可以了。这是入门级的随笔,说的是函数对象的定义,使用,以及与函数指针,成员函数指针的关系。
C++函数对象实质上是一个实现了operator()--括号操作符--的类
下面是函数对象与函数指针的例子:
[cpp] view plain copy print ?
  1. namespace  
  2. {  
  3.     class AddCls  
  4.     {    
  5.     public:    
  6.         int operator()(int a, int b)    
  7.         {    
  8.             return a + b;    
  9.         }    
  10.     };  
  11.   
  12.     int AddFunc(int a, int b)    
  13.     {    
  14.         return a + b;    
  15.     }  
  16. }  
  17.   
  18. void test_addObj()  
  19. {  
  20.     // 定义函数对象   
  21.     AddCls addObj;   
  22.     PRINT_DEBUG(addObj(3, 4);  
  23. }  
  24.   
  25. void test_addFunPtr()  
  26. {  
  27.     typedef int (*FunPtr) (int a, int b);    
  28.     FunPtr addFunPtr = &AddFunc;  
  29.     PRINT_DEBUG(addFunPtr(3, 5));  
  30. }  
namespace
{
class AddCls
{  
public:  
int operator()(int a, int b)  
{  
return a + b;  
}  
};
int AddFunc(int a, int b)  
{  
return a + b;  
}
}
void test_addObj()
{
// 定义函数对象
AddCls addObj; 
PRINT_DEBUG(addObj(3, 4);
}
void test_addFunPtr()
{
typedef int (*FunPtr) (int a, int b);  
FunPtr addFunPtr = &AddFunc;
PRINT_DEBUG(addFunPtr(3, 5));
}
除了定义方式不一样,使用方式可是一样的。
既然C++函数对象与函数指针在使用方式上没什么区别,那为什么要用函数对象呢?很简单, 函数对象可以携带附加数据,而指针就不行了。下面就举个使用附加数据的例子:
[cpp] view plain copy print ?
  1. class less    
  2. {    
  3. public:    
  4.     less(int num):n(num){}    
  5.     bool operator()(int value)    
  6.     {   
  7.         return value < n;    
  8.     }    
  9. private:    
  10.     int n;    
  11. };  
  12. less isLess(10);  
  13. cout << isLess(9) << " " << isLess(12); // 输出 1 0   
class less  
{  
public:  
less(int num):n(num){}  
bool operator()(int value)  
{ 
return value < n;  
}  
private:  
int n;  
};
less isLess(10);
cout << isLess(9) << " " << isLess(12); // 输出 1 0 
另一个例子:
[cpp] view plain copy print ?
  1. const int SIZE = 5;    
  2. int array[SIZE] = { 50, 30, 9, 7, 20};    
  3. // 找到小于数组array中小于10的第一个数的位置     
  4. int * pa = std::find_if(array, array + SIZE, less(10));  
  5. // pa 指向 9 的位置     
  6. // 找到小于数组array中小于40的第一个数的位置     
  7. int * pb = std::find_if(array, array + SIZE, less(40));   
  8. // pb 指向 30 的位置  
const int SIZE = 5;  
int array[SIZE] = { 50, 30, 9, 7, 20};  
// 找到小于数组array中小于10的第一个数的位置  
int * pa = std::find_if(array, array + SIZE, less(10));
// pa 指向 9 的位置  
// 找到小于数组array中小于40的第一个数的位置  
int * pb = std::find_if(array, array + SIZE, less(40)); 
// pb 指向 30 的位置
要想让一个函数既能接受函数指针,也能接受函数对象,最方便的方法就是用模板。如:
[cpp] view plain copy print ?
  1. template<typename FUNC>  
  2. int count_n(int* array, int size, FUNC func)    
  3. {    
  4.   int count = 0;    
  5.   for(int i = 0; i < size; ++i)    
  6.   if(func(array[i]))    
  7.     count ++;    
  8.   return count;    
  9. }  
template<typename FUNC>
int count_n(int* array, int size, FUNC func)  
{  
int count = 0;  
for(int i = 0; i < size; ++i)  
if(func(array[i]))  
count ++;  
return count;  
}
这个函数可以统计数组中符合条件的数据个数,如:
[cpp] view plain copy print ?
  1. const int SIZE = 5;    
  2. int array[SIZE] = { 50, 30, 9, 7, 20};    
  3. cout << count_n(array, SIZE, less(10)); // 2     
  4. // 用函数指针也没有问题:     
  5. bool less10(int v)    
  6. {    
  7.   return v < 10;    
  8. }    
  9. cout << count_n(array, SIZE, less10); // 2   
const int SIZE = 5;  
int array[SIZE] = { 50, 30, 9, 7, 20};  
cout << count_n(array, SIZE, less(10)); // 2  
// 用函数指针也没有问题:  
bool less10(int v)  
{  
return v < 10;  
}  
cout << count_n(array, SIZE, less10); // 2 

这篇关于C++函数对象与函数指针不同之处的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1145864

相关文章

C++中unordered_set哈希集合的实现

《C++中unordered_set哈希集合的实现》std::unordered_set是C++标准库中的无序关联容器,基于哈希表实现,具有元素唯一性和无序性特点,本文就来详细的介绍一下unorder... 目录一、概述二、头文件与命名空间三、常用方法与示例1. 构造与析构2. 迭代器与遍历3. 容量相关4

C++中悬垂引用(Dangling Reference) 的实现

《C++中悬垂引用(DanglingReference)的实现》C++中的悬垂引用指引用绑定的对象被销毁后引用仍存在的情况,会导致访问无效内存,下面就来详细的介绍一下产生的原因以及如何避免,感兴趣... 目录悬垂引用的产生原因1. 引用绑定到局部变量,变量超出作用域后销毁2. 引用绑定到动态分配的对象,对象

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

JavaScript对象转数组的三种方法实现

《JavaScript对象转数组的三种方法实现》本文介绍了在JavaScript中将对象转换为数组的三种实用方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友... 目录方法1:使用Object.keys()和Array.map()方法2:使用Object.entr

Python中isinstance()函数原理解释及详细用法示例

《Python中isinstance()函数原理解释及详细用法示例》isinstance()是Python内置的一个非常有用的函数,用于检查一个对象是否属于指定的类型或类型元组中的某一个类型,它是Py... 目录python中isinstance()函数原理解释及详细用法指南一、isinstance()函数

python中的高阶函数示例详解

《python中的高阶函数示例详解》在Python中,高阶函数是指接受函数作为参数或返回函数作为结果的函数,下面:本文主要介绍python中高阶函数的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录1.定义2.map函数3.filter函数4.reduce函数5.sorted函数6.自定义高阶函数

Python中的sort方法、sorted函数与lambda表达式及用法详解

《Python中的sort方法、sorted函数与lambda表达式及用法详解》文章对比了Python中list.sort()与sorted()函数的区别,指出sort()原地排序返回None,sor... 目录1. sort()方法1.1 sort()方法1.2 基本语法和参数A. reverse参数B.

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

使用MapStruct实现Java对象映射的示例代码

《使用MapStruct实现Java对象映射的示例代码》本文主要介绍了使用MapStruct实现Java对象映射的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、什么是 MapStruct?二、实战演练:三步集成 MapStruct第一步:添加 Mave

C++中处理文本数据char与string的终极对比指南

《C++中处理文本数据char与string的终极对比指南》在C++编程中char和string是两种用于处理字符数据的类型,但它们在使用方式和功能上有显著的不同,:本文主要介绍C++中处理文本数... 目录1. 基本定义与本质2. 内存管理3. 操作与功能4. 性能特点5. 使用场景6. 相互转换核心区别