在C++中使用libuv时对回调的处理 (2)
前情简介
在完成了第一版的《在C++中使用libuv时对回调的处理》之后,在对项目进行开发的时候,还是感觉有一些难受。
因为在实际操作的时候,需要构建一个结构体,并且需要对这个结构体的内存进行管理,非常的麻烦。
在对C++的模板编程进行简单的学习后,了解到一个比较基本的知识。如果一个值或者类型能在编译的时候确定,那么它是一定可以作为模板参数的。
反观我之前为了完成操作构建的结构体,可以很明显的发现,成员函数指针那一个变量是一直保持不变的,而且可以在编译的时候确定,所以是有办法将成员函数指针放入模板里面的。
解决方案
使用回调的函数
typedef struct {
void *data;
int s;
} call_back_t;
typedef void (*call_back_func_t)(call_back_t *t, int s, int v);
int call_back_func(call_back_t *t, call_back_func_t func) {
func(t, t->s, 12);
return 0;
}
回调函数及其类
class CallBack {
public:
int a = 0;
void call_back(call_back_t *t, int s, int v) {
std::cout << "t->s:" << t->s << std::endl;
std::cout << "s:" << s << std::endl;
std::cout << "v:" << v << std::endl;
std::cout << "a:" << a << std::endl;
}
};
解决方案
template<typename T, T>
struct comm_call_back_s;
template<typename ClassType, typename ...ArgTypes, void (ClassType::*FunType)(call_back_t *t, ArgTypes...) >
struct comm_call_back_s<void (ClassType::*)(call_back_t *t, ArgTypes...), FunType> {
static void comm_call_back(call_back_t *t, ArgTypes... Value) {
ClassType *mClass = static_cast<ClassType *>(t->data);
(mClass->*FunType)(t, std::forward<ArgTypes>(Value)...);
}
};
#define define_comm_call_back_s(F) (comm_call_back_s<decltype((F)), (F)>::comm_call_back)
以上代码是根据[1]中大佬代码修改得来的。首先是第一段
template<typename T, T>
struct comm_call_back_s;
这一段代码定义了模板的原型,模板包括两个参数。一个是类型T,另一个是类型为T的值。
template<typename ClassType, typename ...ArgTypes, void (ClassType::*FunType)(call_back_t *t, ArgTypes...) >
struct comm_call_back_s<void (ClassType::*)(call_back_t *t, ArgTypes...), FunType> {
static void comm_call_back(call_back_t *t, ArgTypes... Value) {
ClassType *mClass = static_cast<ClassType *>(t->data);
(mClass->*FunType)(t, std::forward<ArgTypes>(Value)...);
}
};
第二段代码是我们主要使用的偏特化模板。一共定义了三个模板参数,第一个是包含回调函数的类,第二个是回调函数的部分参数,第三个是回调函数。
在具体的特化中,我们将回调函数的类型作为原型里面的类型T,回调函数作为值。
然后,我们定义了一个comm_call_back函数作为我们封装的回调函数。
#define define_comm_call_back_s(F) (comm_call_back_s<decltype((F)), (F)>::comm_call_back)
最后一段,我们定义了一个宏,方便我们的调用。
使用
int main() {
CallBack b;
b.a = 100;
call_back_t call;
call.s = 1024;
call.data = static_cast<void *>(&b);
call_back_func(&call, define_comm_call_back_s(&CallBack::call_back));
}
反思
在写完这一些代码后,我思考了几个问题,并做了一定的解答。
- 为什么使用结构体,而不直接使用模板函数。
因为我们在定义模板原型的时候没办法决定函数的参数,所以先使用结构体定义,然后使用偏特化实现具体的函数。
引用
[1] https://stackoverflow.com/questions/9779105/generic-member-function-pointer-as-a-template-parameter
在C++中使用libuv时对回调的处理 (2)的更多相关文章
- 在C++中使用libuv时对回调的处理
新的解决方法 https://www.cnblogs.com/ink19/p/13768425.html libuv简介 libuv是一个可以跨平台的C语言库,它提供了基于事件的异步IO支持[1].提 ...
- 采用指数退避算法实现ajax请求的重发,全部完成时触发回调函数
目录: 0.Chrome扩展开发(Gmail附件管理助手)系列之〇——概述 1.Chrome扩展开发之一——Chrome扩展的文件结构 2.Chrome扩展开发之二——Chrome扩展中脚本的运行机制 ...
- jQuery中ajax方法无法执行回调函数问题
最近遇到一个问题,发现使用jquery的ajax方法时,回调方法无法执行,而使用$.load()方法时却能正确返回数据.经过长时间调试最终发现是自己粗心大意,原来后台返回的是json数据,而返回的数据 ...
- 在Amazon FreeRTOS V10中使用运行时统计信息
在MCU on Eclipse网站上看到Erich Styger在8月2日发的博文,一篇关于在Amazon FreeRTOS V10中使用运行时统计信息的文章,本人觉得很有启发,特将其翻译过来以备参考 ...
- Object—C 块在函数中作为参数时的分析
暂时对这个有了一些粗浅的理解,记下来一边后面学习时学习,改正. 先举个例子: A类: .h文件: @interface A : NSObject - (void)Paly1:(void (^)(do ...
- js的for循环中出现异步函数,回调引用的循环值总是最后一步的值?
这几天跟着视频学习node.js,碰到很多的异步函数的问题,现在将for循环中出现的异步函数回调值的问题总结如下: 具体问题是关于遍历文件夹中的子文件夹的,for循环包裹异步函数的代码: for (v ...
- 在php中定义常量时,const与define的区别?
问]在php中定义常量时,const与define的区别? [答]使用const使得代码简单易读,const本身就是一个语言结构,而define是一个函数.另外const在编译时要比define快很 ...
- AndRodi Strudio中的按钮时件
AndRodi Studio中的按钮时件注册一定要写在onCraete中 @Override protected void onCreate(Bundle savedInstanceState) { ...
- 在MySQL向表中插入中文时,出现:incorrect string value 错误
在MySQL向表中插入中文时,出现:incorrect string value 错误,是由于字符集不支持中文.解决办法是将字符集改为GBK,或UTF-8. 一.修改数据库的默认字符集 ...
随机推荐
- 网站SEO优化技术转让奇人
http://www.wocaoseo.com/thread-111-1-1.html 本月假期我排到了今天星期二,由于工作性质原因经常会熬夜,养成一种不好的习惯"睡懒觉"视为享受 ...
- QT_QGIS_基本使用
QT_QGIS_基本使用 1.新建画布 2.添加矢量图层 1.打开矢量图层 2.新建矢量图层 1.添加几何要素--点 2.添加几何要素--线 3.添加栅格图层 1.打开栅格图层 小 ...
- SpringBoot中加载XML配置
开篇 在SpringBoot中我们通常都是基于注解来开发的,实话说其实这个功能比较鸡肋,但是,SpringBoot中还是能做到的.所以用不用是一回事,会不会又是另外一回事. 涛锅锅在个人能力能掌握的范 ...
- Copy a Xaml object
<Control.Resources> <Button Click="Button_OnClick" x:Key="MyButton"> ...
- ACboy needs your help (动态规划背包)
ACboy has N courses this term, and he plans to spend at most M days on study.Of course,the profit he ...
- Content Security Policy (CSP)内容安全策略总结
跨域脚本攻击 XSS 是最常见.危害最大的网页安全漏洞. 为了防止它们,要采取很多编程措施,非常麻烦.很多人提出,能不能根本上解决问题,浏览器自动禁止外部注入恶意脚本?这就是"网页安全政策& ...
- MySQL查询更新所有满足条件的数据
-- 将订单表所有的状态改成1update oc_repair_preorder a inner join (select id,`status` from oc_repair_preorder) b ...
- Netty内置的编解码器和ChannelHandler
Netty 为许多通用协议提供了编解码器和处理器,几乎可以开箱即用,这减少了你在那些相当繁琐的事务上本来会花费的时间与精力. 通过SSL/TLS 保护Netty 应用程序 SSL和TLS这样的安全协议 ...
- meta生成器 —— 表单元素组件
手写代码? meta(json)需要手写吗?别闹,手写多麻烦呀,我这么懒怎么可能手写,这辈子都别想,所以要弄个工具出来,咱们说干就干. 这个工具,说白了本身就是一个表单,一个meta属性对应一个met ...
- python中多进程multiprocessing、多线程threading、线程池threadpool
浅显点理解:进程就是一个程序,里面的线程就是用来干活的,,,进程大,线程小 一.多线程threading 简单的单线程和多线程运行:一个参数时,后面要加逗号 步骤:for循环,相当于多个线程——t=t ...