c++回调函数 callback

2024-05-31 05:08
文章标签 c++ 函数 回调 callback

本文主要是介绍c++回调函数 callback,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1Callback方式
Callback
的本质是设置一个函数指针进去,然后在需要需要触发某个事件时调用该方法比如Windows的窗口消息处理函数就是这种类型。比如下面的示例代码,我们在Download完成时需要触发一个通知外面的事件:

  

复制代码
 1 typedef void (__stdcall *DownloadCallback)(const char* pURL, bool bOK);  
 2 void DownloadFile(const char* pURL, DownloadCallback callback)  
 3 {  
 4     cout << "downloading: " << pURL << "" << endl;  
 5     callback(pURL, true);  
 6 }  
 7 void __stdcall OnDownloadFinished(const char* pURL, bool bOK)  
 8 {  
 9     cout << "OnDownloadFinished, URL:" << pURL << "    status:" << bOK << endl;  
10 } 
复制代码

 

(2)Sink方式

Sink的本质是你按照对方要求实现一个C++接口,然后把你实现的接口设置给对方,对方需要触发事件时调用该接口, COM中连接点就是居于这种方式。上面下载文件的需求,如果用Sink实现,代码如下:

 

 

复制代码
 1 class IDownloadSink  
 2 {  
 3 public:  
 4     virtual void OnDownloadFinished(const char* pURL, bool bOK) = 0;  
 5 };  
 6     
 7     
 8 class CMyDownloader  
 9 {  
10 public:  
11     CMyDownloader(IDownloadSink* pSink)  
12         :m_pSink(pSink)  
13     {  
14     }  
15     
16     void DownloadFile(const char* pURL)  
17     {  
18         cout << "downloading: " << pURL << "" << endl;  
19         if(m_pSink != NULL)  
20         {  
21             m_pSink->OnDownloadFinished(pURL, true);  
22         }  
23     }  
24     
25 private:  
26     IDownloadSink* m_pSink;  
27 };  
28     
29 class CMyFile: public IDownloadSink  
30 {  
31 public:  
32     void download()  
33     {  
34         CMyDownloader downloader(this);  
35         downloader.DownloadFile("www.baidu.com");  
36     }  
37     
38     virtual void OnDownloadFinished(const char* pURL, bool bOK)  
39     {  
40         cout << "OnDownloadFinished, URL:" << pURL << "    status:" << bOK << endl;  
41     }  
42 };  
复制代码

 

 

 

(3)Delegate方式
    Delegate
的本质是设置成员函数指针给对方,然后让对方在需要触发事件时调用。C#中用Delegate的方式实现Event,让C++程序员很是羡慕,C++中因为语言本身的关系,要实现Delegate还是很麻烦的。上面的例子我们用Delegate的方式实现如下:

 

 

复制代码
class CDownloadDelegateBase  
{  
public:  virtual void Fire(const char* pURL, bool bOK) = 0;  
};  template<typename O, typename T>  
class CDownloadDelegate: public CDownloadDelegateBase  
{  typedef void (T::*Fun)(const char*, bool);  
public:  CDownloadDelegate(O* pObj = NULL, Fun pFun = NULL)  :m_pFun(pFun), m_pObj(pObj)  {  }  virtual void Fire(const char* pURL, bool bOK)  {  if(m_pFun != NULL  && m_pObj != NULL)  {  (m_pObj->*m_pFun)(pURL, bOK);  }  }  private:  Fun m_pFun;  O* m_pObj;  
};  template<typename O, typename T>  
CDownloadDelegate<O,T>* MakeDelegate(O* pObject, void (T::*pFun)(const char* pURL, bool))  
{  return new CDownloadDelegate<O, T>(pObject, pFun);  
}  class CDownloadEvent  
{  
public:  ~CDownloadEvent()  {  vector<CDownloadDelegateBase*>::iterator itr = m_arDelegates.begin();  while (itr != m_arDelegates.end())  {  delete *itr;  ++itr;  }  m_arDelegates.clear();  }  void operator += (CDownloadDelegateBase* p)  {  m_arDelegates.push_back(p);  }  void operator -= (CDownloadDelegateBase* p)  {  ITR itr = remove(m_arDelegates.begin(), m_arDelegates.end(), p);  ITR itrTemp = itr;  while (itrTemp != m_arDelegates.end())  {  delete *itr;  ++itr;  }  m_arDelegates.erase(itr, m_arDelegates.end());  }  void operator()(const char* pURL, bool bOK)  {  ITR itrTemp = m_arDelegates.begin();  while (itrTemp != m_arDelegates.end())  {  (*itrTemp)->Fire(pURL, bOK);  ++itrTemp;  }  }  private:  vector<CDownloadDelegateBase*> m_arDelegates;  typedef vector<CDownloadDelegateBase*>::iterator ITR;  
};  class CMyDownloaderEx  
{  
public:  void DownloadFile(const char* pURL)  {  cout << "downloading: " << pURL << "" << endl;  downloadEvent(pURL, true);  }  CDownloadEvent downloadEvent;  
};  class CMyFileEx  
{  
public:  void download()  {  CMyDownloaderEx downloader;  downloader.downloadEvent += MakeDelegate(this, &CMyFileEx::OnDownloadFinished);  downloader.DownloadFile("www.baidu.com");  }  virtual void OnDownloadFinished(const char* pURL, bool bOK)  {  cout << "OnDownloadFinished, URL:" << pURL << "    status:" << bOK << endl;  }  
}; 
复制代码

 

 

    可以看到Delegate的方式代码量比上面其他2种方式大多了,并且我们上面是固定参数数量和类型的实现方式,如果要实现可变参数,要更加麻烦的多。可变参数的方式可以参考这2种实现:

Yet Another C#-style Delegate Class in Standard C++
Member Function Pointers and the Fastest Possible C++ Delegates

   

我们可以用下面的代码测试我们上面的实现:

 

 

 

复制代码
int _tmain(int argc, _TCHAR* argv[])  
{  DownloadFile("www.baidu.com", OnDownloadFinished);  CMyFile f1;  f1.download();  CMyFileEx ff;  ff.download();  system("pause");  return 0;  
}  
复制代码

 

 

最后简单比较下上面3种实现回调的方法:


第一种Callback的方法是面向过程的,使用简单而且灵活,正如C语言本身。
第二种Sink的方法是面向对象的,在C++里使用较多,可以在一个Sink里封装一组回调接口,适用于一系列比较固定的回调事件。
第三种Delegate的方法也是面向对象的,和Sink封装一组接口不同,Delegate的封装是以函数为单位,粒度比Sink更小更灵活。

你更倾向于用哪种方式来实现回调?

 

还有一种是通过sigslot信号槽机制来实现委托

 相关sigslot的用法可以参考这篇文章。

相关头文件下载地址

这篇关于c++回调函数 callback的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

基于Java实现回调监听工具类

《基于Java实现回调监听工具类》这篇文章主要为大家详细介绍了如何基于Java实现一个回调监听工具类,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录监听接口类 Listenable实际用法打印结果首先,会用到 函数式接口 Consumer, 通过这个可以解耦回调方法,下面先写一个

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

MySQL高级查询之JOIN、子查询、窗口函数实际案例

《MySQL高级查询之JOIN、子查询、窗口函数实际案例》:本文主要介绍MySQL高级查询之JOIN、子查询、窗口函数实际案例的相关资料,JOIN用于多表关联查询,子查询用于数据筛选和过滤,窗口函... 目录前言1. JOIN(连接查询)1.1 内连接(INNER JOIN)1.2 左连接(LEFT JOI

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve