C++11新特性 变参模板、完美转发(简述)
变参模板 (Variadic Template) - 使得 emplace 可以接受任意参数,这样就可以适用于任意对象的构建
完美转发 - 使得接收下来的参数 能够原样的传递给对象的构造函数,这带来另一个方便性
先来看看下边的代码,看一下C11标准中提供的变参模板的用法之一:
1 #include <iostream>
2 #include <vector>
3
4 using namespace std;
5
6 class student
7 {
8 public:
9 student(int age, const char name[64])
10 {
11 this -> age = age;
12 strncpy_s(this->name, sizeof(this->name) - 1, name, sizeof(name));
13 cout << "调用有参构造函数" << endl;
14 }
15
16 student(const student& stud)
17 {
18 this->age = stud.age;
19 strncpy_s(this->name, sizeof(this->name) - 1, stud.name, sizeof(stud.name));
20 cout << "调用拷贝构造函数" << endl;
21 }
22
23 ~student()
24 {
25 cout << "调用析构函数" << endl;
26 }
27
28 private:
29 int age;
30 char name[64];
31
32 };
33
34 int main()
35 {
36 vector<student> vectStud;
37
38 //方法一:
39 //student stud(18, "姓名");
40 //vectStud.push_back(stud);
41
42 //方法二:
43 vectStud.push_back(student(18, "姓名"));
44
45 return 0;
46 }
方法一 与 方法二 的打印结果完全相同:

打印中第一行的有参构造函数是第39行代码调用到的,40行代码将这个对象放置到vector容器中时,会创建一个临时对象进行拷贝。这会造成效能的浪费。
为了解决这个问题C++11中加入了 变参模板 & 完美转发,使得 emplace 可以接受任意参数,这样就可以适用于任意对象的构建,如下代码:
1 #include <iostream>
2 #include <vector>
3
4 using namespace std;
5
6 class student
7 {
8 public:
9 student(int age, const char name[64])
10 {
11 this -> age = age;
12 strncpy_s(this->name, sizeof(this->name) - 1, name, sizeof(name));
13 cout << "调用有参构造函数" << endl;
14 }
15
16 student(const student& stud)
17 {
18 this->age = stud.age;
19 strncpy_s(this->name, sizeof(this->name) - 1, stud.name, sizeof(stud.name));
20 cout << "调用拷贝构造函数" << endl;
21 }
22
23 ~student()
24 {
25 cout << "调用析构函数" << endl;
26 }
27
28 private:
29 int age;
30 char name[64];
31
32 };
33
34 int main()
35 {
36 vector<student> vectStud;
37
38 //方法一:
39 //student stud(18, "姓名");
40 //vectStud.push_back(stud);
41 //方法二:
42 //vectStud.push_back(student(18, "姓名"));
43
44 //使用 emplace_back
45 vectStud.emplace_back(18, "姓名");
46
47 return 0;
48 }
从打印结果可以看出,使用了 emplace_back 之后可以省略一个零时对象,并且不用进行拷贝构造函数的操作。
emplace_back 也就相当于一个 push_back 的操作

还有一个 emplace ,是在指定位置插入相关数据,类似于 insert,如下代码:
1 #include <iostream>
2 #include <vector>
3
4 using namespace std;
5
6 class student
7 {
8 public:
9 student(int age, const char name[64])
10 {
11 this -> age = age;
12 strncpy_s(this->name, sizeof(this->name) - 1, name, sizeof(name));
13 cout << "调用有参构造函数" << endl;
14 cout << "姓名:" << this->name << " " << "年龄:" << this->age << endl;
15 }
16
17 student(const student& stud)
18 {
19 this->age = stud.age;
20 strncpy_s(this->name, sizeof(this->name) - 1, stud.name, sizeof(stud.name));
21 cout << "调用拷贝构造函数" << endl;
22 }
23
24 ~student()
25 {
26 cout << "调用析构函数" << endl;
27 }
28
29 private:
30 int age;
31 char name[64];
32 };
33
34 int main()
35 {
36 vector<student> vectStud;
37
38 //方法一:
39 //student stud(18, "姓名");
40 //vectStud.push_back(stud);
41 //方法二:
42 //vectStud.push_back(student(18, "姓名"));
43
44 //使用 emplace_back
45 vectStud.emplace_back(18, "张三");
46
47 //使用 emplace
48 vectStud.emplace(vectStud.begin(), 20, "李四");
49
50 return 0;
51 }
打印结果:

这里有个有趣的现象,第5行出现了一个拷贝构造函数,并且调用了3次析构函数。
这是因为 vector 的存储方式决定的,因为 vector 的存储空间是由元素的多少进行变化的,而我们在第48行的代码中使用了 begin() ,在首位置插入,导致内存空间发生了变化。编译器需要将第一个张三拷贝至李四的前方,然后析构掉。如果前方在插入一个,将会把后两个元素再进行拷贝和析构,以此类推。
不同的编译器,对于 vector 新插入元素的拷贝和析构的顺序有可能不同,即使是 VS2015 与 VS2019 在拷贝和析构的顺序上也不相同。
========================================================================================================================
C++11新特性 变参模板、完美转发(简述)的更多相关文章
- [转载] C++11新特性
C++11标准发布已有一段时间了, 维基百科上有对C++11新标准的变化和C++11新特性介绍的文章. 我是一名C++程序员,非常想了解一下C++11. 英文版的维基百科看起来非常费劲,而中文版维基百 ...
- c++学习书籍推荐《深入理解C++11 C++11新特性解析与应用》下载
百度云及其他网盘下载地址:点我 编辑推荐 <深入理解C++11:C++11新特性解析与应用>编辑推荐:C++标准委员会成员和IBM XL编译器中国开发团队共同撰写,权威性毋庸置疑.系统.深 ...
- c++ 11 线程池---完全使用c++ 11新特性
前言: 目前网上的c++线程池资源多是使用老版本或者使用系统接口实现,使用c++ 11新特性的不多,最近研究了一下,实现一个简单版本,可实现任意任意参数函数的调用以及获得返回值. 0 前置知识 首先介 ...
- C++ 11学习和掌握 ——《深入理解C++ 11:C++11新特性解析和应用》读书笔记(一)
因为偶然的机会,在图书馆看到<深入理解C++ 11:C++11新特性解析和应用>这本书,大致扫下,受益匪浅,就果断借出来,对于其中的部分内容进行详读并亲自编程测试相关代码,也就有了整理写出 ...
- C++11新特性之一——Lambda表达式
C++11新特性总结可以参考:http://www.cnblogs.com/pzhfei/archive/2013/03/02/CPP_new_feature.html#section_6.8 C++ ...
- C++11 新特性之智能指针(shared_ptr, unique_ptr, weak_ptr)
这是C++11新特性介绍的第五部分,涉及到智能指针的相关内容(shared_ptr, unique_ptr, weak_ptr). shared_ptr shared_ptr 基本用法 shared_ ...
- C++11新特性— auto 和 decltype 区别和联系
一. auto简介 编程时候常常需要把表达式的值付给变量,需要在声明变量的时候清楚的知道变量是什么类型.然而做到这一点并非那么容易(特别是模板中),有时候根本做不到.为了解决这个问题,C++11新标准 ...
- c++11新特性注意点
本文记录下一些c++11新特性需要注意的方面,供日后参考 一.auto auto可以当成“占位符”,根据右边的类型自动推导出变量的类型.需要注意的是 auto不能解决溢出的问题. auto可以与指针和 ...
- C++11 变长模版和完美转发实例代码
C++11 变长模版和完美转发实例代码 #include <memory>#include <iostream>#include <vector>#include ...
随机推荐
- MOOC JAVA笔记
MOOC JAVA笔记 1.基础了解 JDK是开发人员安装的,它提供了开发java程序的必须工具 JRE是普通用户安装的,它提供了java的运行环境 JVM是java虚拟机运行程序的核心 2.程序的移 ...
- scrapy学习之爬虫练习平台35
前言 上一篇文章中爬取了爬虫练习平台的所有 ssr 网站,都是比较简单的,没有反爬措施,这次来爬一下后面的 spa 系列. 环境准备 这里沿用了上篇文章的环境和设置,就不重新搭建环境了. 开始爬取 s ...
- php递归无限查询上级或者下级
$this->get_array($user['uid'],1); function get_array($user_id,$top=0){ $sql = 'SELECT * FROM ' . ...
- day95:flask:SQLAlchemy数据库查询进阶&关联查询
目录 1.数据库查询-进阶 1.常用的SQLAlchemy查询过滤器 2.常用的SQLAlchemy查询结果的方法 3.filter 4.order_by 5.count 6.limit&of ...
- c#多播委托判断空值
int resualt = (int)fn?.Invoke(a, b); 其中fn是有两个参数一个int返回值的多播委托,这个看起来很精炼,大概意思也不懂 个人理解为使用先判断fn?,再传值以及输出为 ...
- AcWing 298. 围栏 (POJ1821)
标签(空格分隔): dp 单调队列优化 题目描述 有N块木板从左到右排成一行,有M个工匠对这些木板进行粉刷,每块木板至多被粉刷一次. 第 i 个木匠要么不粉刷,要么粉刷包含木板 \(S_i\) 的,长 ...
- 精尽MyBatis源码分析 - MyBatis-Spring 源码分析
该系列文档是本人在学习 Mybatis 的源码过程中总结下来的,可能对读者不太友好,请结合我的源码注释(Mybatis源码分析 GitHub 地址.Mybatis-Spring 源码分析 GitHub ...
- Kafka入门之consumer--rebalance流程
重平衡(rebalance) 旧版本Kafka依托于Zk进行rebalance,新版本consumer使用了Kafka内置的一个全新的组协调协议.对于每个组而言,Kafka的某个broker会被选举为 ...
- How tomcat works(深入剖析tomcat)servlet容器
How tomcat works (5)servlet容器阅读笔记 第四章阅读了tomcat默认连接器的实现,当时connector中的使用的容器是自定义的容器,也是非常之简单奥,一个人就干完了所有的 ...
- 欢天喜地七仙女——UML设计
这个作业的要求在哪里 作业要求 团队名称 欢天喜地七仙女 团队成员 王玮晗.林鑫宇.黄龙骏.陈少龙.何一山.崔亚明.陆桂莺 这个作业的目标 团队一起绘制UML图 作业正文 如下 其它参考文献 见文末 ...