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

相关文章

在java中如何将inputStream对象转换为File对象(不生成本地文件)

《在java中如何将inputStream对象转换为File对象(不生成本地文件)》:本文主要介绍在java中如何将inputStream对象转换为File对象(不生成本地文件),具有很好的参考价... 目录需求说明问题解决总结需求说明在后端中通过POI生成Excel文件流,将输出流(outputStre

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque

C++常见容器获取头元素的方法大全

《C++常见容器获取头元素的方法大全》在C++编程中,容器是存储和管理数据集合的重要工具,不同的容器提供了不同的接口来访问和操作其中的元素,获取容器的头元素(即第一个元素)是常见的操作之一,本文将详细... 目录一、std::vector二、std::list三、std::deque四、std::forwa

C++字符串提取和分割的多种方法

《C++字符串提取和分割的多种方法》在C++编程中,字符串处理是一个常见的任务,尤其是在需要从字符串中提取特定数据时,本文将详细探讨如何使用C++标准库中的工具来提取和分割字符串,并分析不同方法的适用... 目录1. 字符串提取的基本方法1.1 使用 std::istringstream 和 >> 操作符示

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

C++ 各种map特点对比分析

《C++各种map特点对比分析》文章比较了C++中不同类型的map(如std::map,std::unordered_map,std::multimap,std::unordered_multima... 目录特点比较C++ 示例代码 ​​​​​​代码解释特点比较1. std::map底层实现:基于红黑

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程