c++11学习笔记(8)- 可变参数宏、函数、模板


1. 可变参数宏

可变参数的宏在C语言中就有定义,使用 ,在宏定义中表示可变参数;使用 VA_ARGS 替代宏定义中的可变参数。
比如我们想要定义 printf() 函数为宏 PR,下面是一个简单的例子:

// 宏定义PR为printf函数
#ifndef PR
#define PR(pStr, ...) \
	printf(pStr, __VA_ARGS__);
#endif

int main(int argc, char** argv){
	PR("%d, %d\n", 10, 20)
}

函数的运行结果为:
10,20

2. 可变参数函数

下面是一个求和的一个函数

int getSum(int count, ...){
	va_list ap;
	int sum = 0;
	va_start(ap, count);
	for (int i = 0; i < count; ++i)
		sum += va_arg(ap, int);
	va_end(ap);
	return sum;
}

int main(int argc, char** argv) {
	std::cout << getSum(3, 1, 2, 3) << std::endl;
	std::cout << getSum(5, 1, 2, 3, 4, 5) << std::endl;
}

上面程序运行结果为:
6
15

这是一个求和函数,函数中count表示参数的个数;va_list 类型的ap用来辅助获取参数。这里首先使用函数 va_start() 对ap进行初始化,使ap变成了一个边长参数的句柄;然后使用函数 va_arg() 将参数一一取出进行运算;最后使用函数 va_end() 结束操作。

3. 可变参数模板

使用可变参数模板时,在 typename 或者 class 后面加上三个点(…)表示可变参数模板。使用sizeof… 可以获取参数的个数; 下面是一个简单的例子

template<typename... T>
void f(T... args){
	std::cout << sizeof...(args) << std::endl;
}

int main(int argc, char** argv) {
	f(1, 2);
	f(1, 2, 3);
	f(1, 2, "aaa", 20.5);
}

程序的输出结果为
2
3
4

(1) 可变参数模板的展开 - 递归展开

关于可变参数模板的展开有两种方式,一种是递归形式的展开;必须要定义一个终止函数,如下面的程序:

template<typename T>
void func(T t){	// 终止函数
	std::cout << t << std::endl;
}

template<typename T, typename... Args>
void func(T head, Args... args){
	std::cout << head << std::endl;
	func(args...);
}

int main(int argc, char** argv){
	func(1, 2, 3, 4, 5);
}

终止函数是为了递归终止时使用的,上面的代码展开为:
func(1, 2, 3, 4, 5);
func(2, 3, 4, 5);
func(3, 4, 5);
func(4, 5);
当递归到func(4, 5)时,head传递4, 调用已经具化的函数func(5)。

可以简单写一个求和的模板函数:

template<typename T>
T sum(T t){
	return t;
}

template<typename T, typename... Args>
T sum(T head, Args... args){
	return head + sum(args...);
}

int main(int argc, char** argv){
	std::cout << sum(1, 2, 3, 4, 5) << std::endl;
}

上面的的程序运行结果为:
15

(2) 可变参数模板的展开 - 非递归展开

可以使用初始化参数列表的形式展开函数模板, 首先看一下下面的代码:

template<typename T>
void printTag(T t){
	std::cout << t << std::endl;
}

template<typename... Args>
void expand(Args... args){
	int array[] = { (printTag(args), 0)... };
}

int main(int argc, char** argv){
	expand(1, 2, 3, 4, 5);
}

上面的代码展开为:
int array[] = { (printTag(1), 0), (printTag(2), 0), (printTag(3), 0), (printTag(4), 0), (printTag(5), 0) };

同理也可以使用初始化参数列表的形式:

template<typename Func, typename... Args>
void expandFunc(const Func& f, Args... args){
	std::initializer_list<int>{(f(std::forward<Args>(args)), 0)...};
}

int main(int argc, char** argv) {
	expandFunc([](int i){std::cout << i << std::endl; }, 1, 2, 3);
}

这里使用的 std::forward 实现完美转发,关于完美转发可参考这篇文章
c++11学习笔记(5)- 引用折叠和完美转发
lambda表达式传递Func,参数1,2,3分别调用该lambda表达式。

(3) 一个函数代理类的实例

我们要实现一个函数代理类,这个类的用法如下:

class TestFuncObject
{
public:
	int runFunc1(int a, int b){
		return a + b;
	}
	int runFunc2(int a, int b, int c){
		return a + b + c;
	}
}

int main(int argc, char** argv){
	TestFuncObject o;
	
	auto  f1 = createDelegate(&o, &TestFuncObject::runFunc1);
	std::cout << f1(10, 20) << std::endl;
	
	auto  f2 = createDelegate(&o, &TestFuncObject::runFunc2);
	std::cout << f2(10, 20, 30) << std::endl;
}

函数 createDelegate 创建一个代理类,然后直接使用该代理函数就可完成被代理函数。
函数的运行结果为:
30
50

下面时代理函数和代理类的具体实现:

template<typename T, typename R, typename... Args>
class MyDelegate{
public:
	MyDelegate(T* t, R (T::*f)(Args...)) :m_t(t), m_f(f){}
	
	R operator()(Args... args){
		return (m_t->*m_f)(std::forward<Args>(args)...);
	}

private:
	T* m_t;
	R (T::*m_f)(Args...);
};

template<typename T, typename R, typename... Args>
MyDelegate<T, R, Args...> createDelegate(T* t, R(T::*f)(Args...)){
	return MyDelegate<T, R, Args...>(t, f);
}

MyDelegate 为代理类,重载了(),为一个仿函数。m_t为对象指针,m_f为该对象的函数指针。createDelegate 为代理函数。


作者:douzhq
个人主页:https://www.douzhq.cn
文章同步页: https://douzhq.cn/c11_8/

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值