C++ 11 std::bind函数绑定如何省略占位参数std::placeholders::_1……-程序员宅基地

技术标签: # C/C++  C++  bind  占位  编程语言  C++ 11  placeholders::_1  

我们在使用C++的函数绑定功能时,每次都需要写一大堆的占位符,有多少个参数就要写多少个占位符,比如下面的示例代码:

class CTest
{
public:
	void foo1(int a, string& b, float c, double d, void* p);
	void foo2(int a, string& b, float c, double d, void* p, short f);
};

int main()
{
	CTest t;
	// 由于CTest::foo1有5个参数,所以这里使用函数绑定时需要写5个占位符
	auto f1 = std::bind(&CTest::foo1, &t, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3,std::placeholders::_4, std::placeholders::_5);
	// 由于CTest::foo1有6个参数,所以这里使用函数绑定时需要写6个占位符
	auto f2 = std::bind(&CTest::foo2, &t, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3,std::placeholders::_4, std::placeholders::_5, std::placeholders::_6);
	return 0;
}

每个绑定都需要根据函数的参数个数写对应个占位参数,个人感觉很麻烦,所以想省略这些占位参数。根据C++ 11的模板参数允许可变的特性,我写了如下一段代码,以自动推导参数个数并且自动填充,甚是方便。独乐乐不如众乐乐,分享出来共大家使用:

#if _MSC_VER
#define PlaceHolder std::_Ph
#else
#define PlaceHolder std::_Placeholder
#endif

template<int N, int...I>
struct MakeSeqs : MakeSeqs<N - 1, N - 1, I...> {};

template<int...I>
struct MakeSeqs<1, I...>
{
	template<typename T, typename R, typename...Args>
	static auto bind(T* obj, R(T::*_Func)(Args...)) -> decltype(std::bind(_Func, obj, PlaceHolder<I>{}...))
	{
		return std::bind(_Func, obj, PlaceHolder<I>{}...);
	}
};

template <typename T, typename R, typename...Args>
auto Bind(T* t, R(T::*f)(Args...)) -> decltype(MakeSeqs<sizeof...(Args)+1>::bind(t, f))
{
	return MakeSeqs<sizeof...(Args)+1>::bind(t, f);
}

template <typename T, typename R, typename...Args>
auto Bind(R(T::*f)(Args...), T* t) -> decltype(MakeSeqs<sizeof...(Args)+1>::bind(t, f))
{
	return MakeSeqs<sizeof...(Args)+1>::bind(t, f);
}

由于C++11的auto不能完全自动推导类型,所以auto返回类型的函数后面需要写decltype。如果使用C++ 14则更简洁:

#if _MSC_VER
#define PlaceHolder std::_Ph
#else
#define PlaceHolder std::_Placeholder
#endif

template<int N, int...I>
struct MakeSeqs : MakeSeqs<N - 1, N - 1, I...> {};

template<int...I>
struct MakeSeqs<1, I...>
{
	template<class T, class _Fx>
	static auto bind(T&& obj, _Fx&& _Func)
	{
		return std::bind(std::forward<_Fx>(_Func), std::forward<T>(obj), PlaceHolder<I>{}...);
	}
};

template <class T, class R, typename...Args>
auto Bind(T* t, R(T::*f)(Args...))
{
	return MakeSeqs<sizeof...(Args)+1>::bind(t, f);
}

template <class T, class R, typename...Args>
auto Bind(R(T::*f)(Args...), T* t)
{
	return MakeSeqs<sizeof...(Args)+1>::bind(t, f);
}

有了这段辅助代码,前面的示例代码的main函数就可以改写成:

int main()
{
	CTest t;
	auto f1 = Bind(&CTest::foo1, &t);
	auto f2 = Bind(&CTest::foo2, &t);
	return 0;
}

是不是感觉清爽多了?
以上代码在VS2015以及GCC4.9.3以上版本编译通过,GCC需要使用-std=c++11或者-std=c++14参数进行编译。

祝好!

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/witton/article/details/98221038

智能推荐

STM32芯片的ADC引脚的识别与选择_adc12_in4-程序员宅基地

文章浏览阅读1w次,点赞4次,收藏26次。1、以STM32F103ZET6芯片为例,ADC引脚分布为:2、可以看出,一共有IN0--IN15,16个ADC通道,通道IN0--IN9与芯片的其他功能复用了引脚。通道IN10--IN15是ADC独立的引脚。 且ADC1和ADC2有IN0--IN15 16个ADC通道。 而ADC3只有IN0、IN1、IN2、IN3、IN10、IN11、IN12、IN13通道。 ADC123_IN0表示ADC1、ADC2、ADC3都有通道IN0。 ADC12_IN4表示只有A..._adc12_in4

webpack test-程序员宅基地

文章浏览阅读1.3k次。webpack test_webpack test

java批量修改文件名后缀_java修改文件后缀名-程序员宅基地

文章浏览阅读1.4k次。java批量修改文件后缀名_java修改文件后缀名

jeecgboot按字段分表_jeecgboot 分表-程序员宅基地

文章浏览阅读1.3k次。在jeecgboot 使用mybatis框架,mybatis-plus3.1.1后支持分表,以下是在jeecg项目中实行分表一、修改mybatis-plus版本后,必须要在3.1.1版本后 <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-..._jeecgboot 分表

java字符串_gags69zzww-程序员宅基地

文章浏览阅读353次。一、字符串的特性:String 被声明为 final,因此它不可被继承。二、不同JDK版本中String的区别:①: 在 Java 8 中,String 内部使用 char 数组存储数据。②: 在 Java 9 之后,String 类的实现改用 byte 数组存储字符串,同时使用 coder 来标识使用了哪种编码。value 数组被声明为 final,这意味着 value 数组初始化之..._gags69zzww

虚拟机在PE系统下看不到硬盘_pe虚拟系统盘-程序员宅基地

文章浏览阅读9.2k次。故障说明:想在VM虚拟化系统里进入PE重置密码,结果PE系统看不到一个硬盘,空空荡荡的。解决方法:先在虚拟机设置里面看看SCSI控制器的当前类型,如果是准虚拟的情况下,就更改SAS类型即可更改SAS类型最后重新进入PE即可看到硬盘,更改这个参数并不会影响系统。..._pe虚拟系统盘

随便推点

航空航天数值分析matlab算法,数值分析 matlab-程序员宅基地

文章浏览阅读501次。"本书介绍数值方法的基本理论和计算方法,并讲述如何利用MATLAB软件实现各种数值算法。它的突出特点是把经典的数值方法内容与现代MATLAB计算软件相结合,书中每个概念均以实例说明,同时还包含大量习题与编程练习,通过这些实例提高读者的实践能力,并加深对数值计算方法的理解。本书经过缩编后适合一学期课"¥63.96定价:¥487.90(1.32折)1. 精心提炼数百个典型实例,全部源于作者的实际工作和..._数值分析在航空领域的应用

Zookeeper权限管理与Quota管理_zookeeper ip 和quota ip-程序员宅基地

文章浏览阅读598次。1 Zookeeper ACLZooKeeper的权限管理亦即ACL控制功能通过Server、Client两端协调完成:Server端:一个ZooKeeper的节点(znode)存储两部分内容:数据和状态,状态中包含ACL信息。创建一个znode会产生一个ACL列表,列表中每个ACL包括:l 验证模式(scheme)l 具体内容(Id)(当scheme=“digest”时,Id为用户名密码,例..._zookeeper ip 和quota ip

ocr数据集:图片数据集_ocr 图片集-程序员宅基地

文章浏览阅读684次。数据集数据集总数量为370万张图片图片数据集文件介绍测试数据集_ocr 图片集

python识图自动化_聊聊 Python 自动化截图的一些经验-程序员宅基地

文章浏览阅读357次。前言今天想先给大家分享 1 个小白用户的 Airtest 从入门到放弃的故事小 A 是一个自动化的小白,在逛测试论坛的时候,偶然间发现了 Airtest 这个基于图像识别的 UI 自动化框架出于好奇,小 A 试用了这个框架,发现只需要几条简单的截图脚本,就可以对设备进行各种自动化操作,于是小 A 成功种草了这个框架但几天之后,随着小 A 的深入使用,他发现截图脚本并不是他想象中那么“完美”;有时..._python识图框架

python语言的关键字是什么_Python语言的关键字有哪些特点?关键字列举-程序员宅基地

文章浏览阅读272次。在Python中,具有特殊功能的标识符称为关键字。关键字是Python语言自己已经使用的了,不允许开发者自己定义和关键字相同名字的标识符。Python中的关键字如下所示:FalseawaitelseimportpassNonebreakexceptinraiseTrueclassfinallyisretu..._python语言的关键字有哪些

pandas数据处理之groupby的常用用法_pandas怎么处理groupby的结果-程序员宅基地

文章浏览阅读2.8k次。groupby的数据处理简单用法groupby(by=None, axis=0, level=None, as_index=True, sort=True, group_keys=True, squeeze=False, **kwargs)1、by:mapping, function, str, or iterable。用于确定gro..._pandas怎么处理groupby的结果

推荐文章

热门文章

相关标签