【C++】C++11可变参数模板

这篇具有很好参考价值的文章主要介绍了【C++】C++11可变参数模板。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【C++】C++11可变参数模板,C++,c++,开发语言

👀樊梓慕:个人主页

 🎥个人专栏:《C语言》《数据结构》《蓝桥杯试题》《LeetCode刷题笔记》《实训项目》《C++》《Linux》《算法》

🌝每一个不曾起舞的日子,都是对生命的辜负


目录

前言

可变参数模板的定义方式

可变参数模板的使用 

编译时递归展开参数包

可变参数模板的应用:emplace系列函数

对比emplace_back与push_back

emplace系列真正的优势在于浅拷贝的类

总结

List类增添模拟实现emplace系列函数

构造

emplace_back()

emplace()


前言

其实我们之前经常使用可变参数模板,C语言的printf函数大家一定非常熟悉,其实这就是一种可变参数模板:

【C++】C++11可变参数模板,C++,c++,开发语言

那么在C++11引入可变参数模板的设计可以带来什么变化呢?让我们一起来学习下吧! 


欢迎大家📂收藏📂以便未来做题时可以快速找到思路,巧妙的方法可以事半功倍。 

=========================================================================文章来源地址https://www.toymoban.com/news/detail-846316.html

GITEE相关代码:🌟樊飞 (fanfei_c) - Gitee.com🌟

=========================================================================


可变参数模板的定义方式

template<class ...Args>
返回类型 函数名(Args... args)
{
  //函数体
}

例如:

// Args是一个模板参数包,args是一个函数形参参数包
// 声明一个参数包Args...args,这个参数包中可以包含0到任意个模板参数。
template <class ...Args>
void ShowList(Args... args)
{}
  • 模板参数Args前面有『 省略号』,代表它是一个可变模板参数,我们把带省略号的参数称为参数包,参数包里面可以包含0到任意个模板参数,而args则是一个函数形参参数包。
  • 模板参数包Args和函数形参参数包args的名字可以任意指定,并不是说必须叫做Args和args,判断是否为参数包的主要关键在『 省略号』。

可变参数模板的使用 

此时我们可以传入任意多个参数了,并且这些参数可以是不同类型的:

int main()
{
    ShowList();
    ShowList(1);
    ShowList(1, 'A');
    ShowList(1, 'A', string("hello"));
    return 0;
}

我们可以在函数模板中通过sizeof计算参数包中参数的个数:

template<class ...Args>
void ShowList(Args... args)
{
    cout << sizeof...(args) << endl; //获取参数包中参数的个数
}

但是,我们如何解析参数包中的内容呢?

我们可不可以这样获取?

template<class ...Args>
void ShowList(Args... args)
{

    for (int i = 0; i < sizeof...(args); i++)
    {
        cout << args[i] << " ";
    }
    cout << endl;
}

答案是不可以!

注意:可变参数模板,既然是模板就是编译时解析,就不能使用如上这种运行时解析的逻辑获取。

因此要获取参数包中的各个参数,可以通过『 编译时递归』的方式解析数据。


编译时递归展开参数包

如何实现编译时递归呢?那肯定是利用编译器的解析机制,我们给函数模板增加一个模板参数,每次从接收到的参数包中剥离出来一个参数,然后在函数模板中递归调用该函数模板,调用时传入剩下的参数包,如此递归下去,每次剥离出参数包中的一个参数,直到参数包中的所有参数都被取出来。

比如:

//展开函数
template<class T, class ...Args>
void _ShowList(T value, Args... args)
{
	cout << value << " "; 
	_ShowList(args...);//递归
}

那么如何终止递归呢?

我们每次都剥离下一个参数,最后必然就没有参数了,那么根据编译器的『最匹配原则 』,我们可以实现一个『 无参』的递归终止函数:

//递归终止函数
void _ShowList()
{
	cout << endl;
}
//展开函数
template<class T, class ...Args>
void _ShowList(T value, Args... args)
{
	cout << value << " ";
	_ShowList(args...);    //递归
}

然后再封装起来如下:

//递归终止函数
void _ShowList()
{
	cout << endl;
}
//展开函数
template<class T, class ...Args>
void _ShowList(T value, Args... args)
{
	cout << value << " "; //打印传入的若干参数中的第一个参数
	_ShowList(args...); //将剩下参数继续向下传
}
//供外部调用的函数
template<class ...Args>
void ShowList(Args... args)
{
	_ShowList(args...);
}

这种『 编译时递归』的思想可谓是非常新奇,值得我们学习。


可变参数模板的应用:emplace系列函数

对比emplace_back与push_back

【C++】C++11可变参数模板,C++,c++,开发语言

【C++】C++11可变参数模板,C++,c++,开发语言

还记得么?

&&这里为万能引用,不是单纯的右值引用。

 相对于push_back,emplace_back支持万能引用和可变参数模板。

他们都是尾插『 一个』数据,注意这里不要看可变参数模板就以为是插入几个值,这里是『 类型』。

对比push_back与emplace_back:

int main()
{
	std::list<pair<F::string, F::string>> lt2;
	pair<F::string, F::string> kv1("xxxx", "yyyy");
	lt2.push_back(kv1);
	lt2.push_back(move(kv1));
	cout << "=============================================" << endl;

	pair<F::string, F::string> kv2("xxxx", "yyyy");
	lt2.emplace_back(kv2);
	lt2.emplace_back(move(kv2));
	cout << "=============================================" << endl;

	return 0;
}

对比发现也没有区别?

【C++】C++11可变参数模板,C++,c++,开发语言

其实emplace_back和push_back真正的区别在于:

push_back需要先用参数构造pair这个对象,然后再将这个对象拷贝给链表节点中的pair;

而emplace_back是直接拿着参数去构造链表节点中的pair,中间省略了拷贝的过程;

比如:

int main()
{
	std::list<pair<F::string, F::string>> lt2;
	pair<F::string, F::string> kv1("xxxx", "yyyy");
	lt2.push_back(kv1);
	lt2.push_back(move(kv1));
	cout << "=============================================" << endl;

	pair<F::string, F::string> kv2("xxxx", "yyyy");
	lt2.emplace_back(kv2);
	lt2.emplace_back(move(kv2));
	cout << "=============================================" << endl;

	lt2.emplace_back("xxxx", "yyyy");
	cout << "=============================================" << endl;
	return 0;
}

【C++】C++11可变参数模板,C++,c++,开发语言


emplace系列真正的优势在于浅拷贝的类

因为对于深拷贝的且实现了移动构造的类来说,移动构造代价很小,emplace的优势显现不出来。

比如:

int main()
{
	std::list<F::string> lt1;
	lt1.push_back("xxxx");

	cout << "=============================================" << endl;

	lt1.emplace_back("xxxx");
	return 0;
}

【C++】C++11可变参数模板,C++,c++,开发语言

emplace真正的优势在于浅拷贝的类,可以节省一个拷贝过程:

比如日期类:

class Date
{
public:
	//构造
	Date(int year, int month, int day)
		:_year(year)
		, _month(month)
		, _day(day)
	{
		cout << "Date(int year, int month, int day)" << endl;
	}
	//拷贝构造
	Date(const Date& d)
		:_year(d._year)
		, _month(d._month)
		, _day(d._day)
	{
		cout << "Date(const Date& d)" << endl;
	}
private:
	int _year = 1;
	int _month = 1;
	int _day = 1;
};

同样尾插:

int main()
{
	std::list<Date> lt1;
	lt1.push_back({ 2024,3,30 });
	cout << "=============================================" << endl;
	lt1.emplace_back(2024, 3, 30);
	return 0;
}

【C++】C++11可变参数模板,C++,c++,开发语言

注意:push_back支持initializer_list作为参数是因为push_back的参数就是模板类型,所以他知道插入的对象是Date类型,就直接走多参数的隐式类型转换构造一个Date对象了,而emplace不支持initializer_list作为构造参数,因为emplace需要可变参数模板去底层构造list节点,你放到initializer_list中就相当于把参数又封装起来了:

int main()
{
	std::list<Date> lt1;
	lt1.push_back({ 2024,3,30 });

	// 不支持
	//lt1.emplace_back({ 2024,3,30 });

	// 正确写法
	lt1.emplace_back(2024, 3, 30);

	return 0;
}

如果给emplace的参数是现成的对象(不管有名对象还是匿名对象),那emplace就没有任何优势了:

int main()
{
	std::list<Date> lt1;

	Date d1(2023, 1, 1);
	lt1.push_back(d1);
	lt1.emplace_back(d1);

	cout << "=============================================" << endl;

	lt1.push_back(Date(2023, 1, 1));
	lt1.emplace_back(Date(2023, 1, 1));
	return 0;
}

【C++】C++11可变参数模板,C++,c++,开发语言


总结

  • emplace系列接口使用需要直接传入参数包才能体现emplace接口的价值与意义,因为emplace系列接口真正高效的情况是传入参数包的时候,直接通过参数包构造出对象,避免了中途的一次拷贝。
  • 所以emplace系列接口并不是在所有场景下都比原有的插入接口高效,如果传入的是对象(不管有名还是匿名),那么emplace系列接口的效率其实和原有的插入接口的效率是一样的。
  • 以上两条告诉我们:以后使用emplace就直接传参数包,不要构造了对象后再将该对象作为参数传给emplace,直接传参数包才是emplace存在的价值和意义
  • 并且emplace系列接口对于深拷贝的且实现了移动构造的类意义不大,因为移动构造的代价很小,emplace带来的效率提升并不会很明显,emplace系列接口对于浅拷贝的类可以节省拷贝(拷贝代价高,并且浅拷贝的类没有移动构造),所以emplace对于浅拷贝的类插入提升很明显。

List类增添模拟实现emplace系列函数

构造

template<class ...Args>
ListNode(Args&&... args)
    : _next(nullptr)
    , _prev(nullptr)
    , _data(forward<Args>(args)...)
{}

emplace_back()

template<class ...Args>
void emplace_back(Args&&... args)
{
    emplace(end(), forward<Args>(args)...);
}

emplace()

template<class ...Args>
iterator emplace(iterator pos, Args&&... args)
{
    Node* cur = pos._node;
    Node* prev = cur->_prev;
    Node* newnode = new Node(forward<Args>(args)...);

    prev->_next = newnode;
    newnode->_prev = prev;
    newnode->_next = cur;
    cur->_prev = newnode;

    //return iterator(newnode);
    return newnode;
}

=========================================================================

如果你对该系列文章有兴趣的话,欢迎持续关注博主动态,博主会持续输出优质内容

🍎博主很需要大家的支持,你的支持是我创作的不竭动力🍎

🌟~ 点赞收藏+关注 ~🌟

=========================================================================

到了这里,关于【C++】C++11可变参数模板的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【C++11】移动赋值 | 新的类功能 | 可变参数模板

    C++11中,string中的operator= 包含 参数为右值的版本 C++98中 没有移动赋值和移动构造 , 只有参数为左值 的赋值重载(operator=)和拷贝构造 本来只有两次深拷贝,但是由于调用拷贝赋值时,内部又进行一次拷贝构造,所以导致最终进行三次深拷贝 这里编译器是不能优化的, 因为优

    2024年02月08日
    浏览(45)
  • C++11『右值引用 ‖ 完美转发 ‖ 新增类功能 ‖ 可变参数模板』

    ✨个人主页: 北 海 🎉所属专栏: C++修行之路 🎃操作环境: Visual Studio 2022 版本 17.6.5 自从C++98以来,C++11无疑是一个相当成功的版本更新。它引入了许多重要的语言特性和标准库增强,为C++编程带来了重大的改进和便利。C++11的发布标志着C++语言的现代化和进步,为程序员

    2024年02月05日
    浏览(49)
  • C++11可变参数模板(typename... Args模板参数包或class... Args)(Args... args函数参数包)(递归展开与参数包展开(只支持C++17))

    C++可变参数是指函数的参数个数是可变的,可以在函数定义时不确定参数的个数,需要在函数体内通过特定的语法来处理这些参数。C++11标准引入了新的可变参数模板,使得可变参数的处理更加方便和灵活。在函数定义时,可以使用省略号(…)来表示可变参数,然后通过va_li

    2024年02月08日
    浏览(44)
  • 【C++】C++11新特性重点:可变参数+lambda

    C++11新特性第二篇重点 文章目录 上一篇的补充 一、可变参数模板 二、lambda函数 总结 上一篇我们重点讲解了右值引用+移动语义,关于移动构造和移动赋值还有一些需要补充的知识: 如果你没有自己实现移动构造函数,且没有实现析构函数 、拷贝构造、拷贝赋值重载中的任

    2024年02月09日
    浏览(52)
  • 【C++】C++11右值引用|新增默认成员函数|可变参数模版|lambda表达式

    在C++11之前,我们只有引用的概念,没有接触到所谓的左值引用或者是右值引用这种概念,从C++11开始,增加了右值引用的概念,那么现在我们将对引用进行一个概念上的区分。在此之前我们所说的引用都是左值引用,对于左值引用相关的内容,可以去看一看博主之前写的文章

    2024年02月15日
    浏览(58)
  • 【C】【C++】可变参数、不定参函数的使用

    C 语言中的可变参数写法: ... 1.1 可变宏函数 以日志举例,我们写入日志时只需要输入关键信息,行号文件等由宏函数补全 这其中,我们需要输入的信息是格式不定的,需要用到可变参数 输出结果: C语言库中的宏 __FILE__ :字符串,记录当前文件名 __LINE__ :整型,记录当前

    2024年02月09日
    浏览(39)
  • C++模板的模板参数(五)

    在C++中,模板的模板参数(Template Template Parameters)是一种特殊的模板参数,允许我们将另一个模板作为模板参数传递给一个模板。这种技术可以用于实现更灵活和通用的模板设计。 模板的模板参数使用两个 “template” 来指示,其中第一个 “template” 用于声明模板参数

    2024年02月10日
    浏览(35)
  • 【c++随笔08】可变参数——va_list、va_start、va_end、va_arg

    原创作者:郑同学的笔记 原创地址:https://zhengjunxue.blog.csdn.net/article/details/131690070 qq技术交流群:921273910 当你在编写 C++ 函数时,有时候你会需要处理可变数量的参数。C++ 中提供了 头文件,其中包含了用于处理可变参数的函数和宏。本教程将向你介绍如何使用 来编写可变参

    2024年02月07日
    浏览(32)
  • 【C++】模板进阶——非类型模板参数、模板特化、模板分离编译

    模板参数分为类型形参 与 非类型形参 类型形参 :出现在模板参数列表中,跟在class或者typedename之类的参数类型名称。 非类型形参 :用一个常量作为类(函数)模板的一个参数,在类(函数)模板中可将该参数当成常量来使用。 非类型模板参数的优势: 有些容器需要在创建对象

    2024年02月01日
    浏览(51)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包