【C++ 学习 ㉘】- 详解 C++11 的列表初始化

目录

一、C++11 简介

二、列表初始化

2.1 - 统一初始化

2.2 - 列表初始化的使用细节

2.2.1 - 聚合类型的定义

2.2.2 - 注意事项

2.3 - initializer_list

2.3.1 - 基本使用

2.3.2 - 源码剖析


 


一、C++11 简介

  1. 1998 年,C++ 标准委员会发布了第一版 C++ 标准,即 C++98 标准,并计划以后每 5 年视实际需要更新一次标准。

    所谓标准,即明确 C++ 代码的编写规范,所有的 C++ 程序员都应遵守此标准

  2. 2003 年,C++ 标准委员会发布了第二版 C++ 标准,即 C++03 标准,但由于 C++03 仅修复了一些 C++98 中存在的漏洞,并未修改核心语法,因此人们习惯将这两个标准合称为 C++98/03 标准。

  3. 2011 年,C++ 标准委员会发布了第三版 C++ 标准,即 C++11 标准,相比 C++03,C++11 带来了数量可观的变化,其中包含了约 140 个新特性,以及对 C++03 中约 600 个缺陷的修正,这使得 C++11 更像从 C++98/03 中孕育出来的一种新语言。。

    C++ 标准委员会一开始是计划在 2007 年发布第三版 C++ 标准,即 C++07 标准,但在 2006 年时,标准委员会认为到 2007 年,甚至到 2008 年,都可能无法发布第三版 C++ 标准,所以干脆将第三版 C++ 标准命名为 C++0x,即计划在二十一世纪的第一个 10 年的某个时间发布,但最终直到 2011 年才发布第三版 C++ 标准


二、列表初始化

在 C++98/03 中,对象的初始化方式有很多种,这些不同的初始化方式都有各自的适用范围和作用,没有一种方式可以通用于所有情况。为了统一初始化方式,并且让初始化行为具有确定的效果,C++11 提出了列表初始化的概念

2.1 - 统一初始化

在 C++98/03 中,对于普通数组和可以直接进行内存拷贝(memcpy)的对象,可以使用列表初始化来初始化数据。

int arr[5] = { 0, 1, 2, 3, 4 };
​
struct Point
{int _x;int _y;
} p = { 0, 0 };

在 C++11 中,初始化列表的适用性被大大地增加了,它现在可以适用于任何类型对象的初始化

注意:在 C++11 中,使用列表初始化时,可以添加等号(=),也可以不添加等号

class A
{
public:A(int i = 0) : _i(i) { }
private:A(const A& a) : _i(a._i) { }
private:int _i;
};
​
int main()
{A a1(10);A a2 = 10;A a3 = { 10 };A a4{ 10 };return 0;
}
  1. a3、a4 使用了 C++11 的列表初始化来初始化对象,效果如同 a1 的直接初始化

  2. 至于 a2,10 会通过隐式类型转换调用构造函数 A(int i = 0) 构造出一个匿名对象,然后通过这个匿名对象调用拷贝构造函数 A(const A& a) 构造出 a2,但由于拷贝构造函数是私有的(private),所以编译器会报错

    注意:Linux 中的 g++ 编译器会报错,VS 中的编译器则不会报错

使用 new 操作符创建新对象时也可以使用列表初始化来初始化对象

int* p = new int{ 0 };
int* arr = new int[5]{ 0, 1, 2, 3, 4 };

除了上面所述的内容,列表初始化还可以直接用在函数传参和返回值上

#include <iostream>
#include <string>
using namespace std;
​
class Person
{
public:Person(int id, string name) : _id(id), _name(name){cout << _id << ":" << _name << endl;}
private:int _id;string _name;
};
​
void func1(Person p) { }
​
Person func2() { return { 2, "李四" }; }
​
int main()
{func1({ 1, "张三" });  // 1:张三Person p = func2();  // 2:李四return 0;
}

 

2.2 - 列表初始化的使用细节

在 C++11 中,列表初始化的使用范围被大大地增加了,但一些模糊的概念也随之而来。

#include <iostream>
using namespace std;
​
struct T1
{int _x;int _y;
} t1{ 520, 520 };
​
struct T2
{int _x;int _y;
​T2(int, int) : _x(1314), _y(1314) { }
} t2{ 520, 520 };
​
int main()
{cout << t1._x << ", " << t1._y << endl;  // 520, 520cout << t2._x << ", " << t2._y << endl;  // 1314, 1314return 0;
}

在上面的程序中,t1 和 t2 都使用相同的列表初始化来初始化对象,但输出的结果却不同。因为对于聚合类型的对象 t1,它可以直接使用列表初始化来初始化对象;对于非聚合类型的对象 t2,它是基于构造函数使用列表初始化来初始化对象

2.2.1 - 聚合类型的定义

  1. 普通数组可以看作是一个聚合类型。

  2. 满足以下条件的类(class、struct、union)可以看作是一个聚合类型:

    • 无基类、无虚函数以及无用户自定义的构造函数

    • 无 private 或 protected 的普通数据成员(即非静态数据成员)

      struct T1
      {int _x;int _y;
      private:  // 或者 protectedint _z;
      } t1{ 1, 2, 3 };  // error(类中有私有成员,无法使用列表初始化进行初始化)
      ​
      struct T2
      {int _x;int _y;
      protected:  // 或者 protectedstatic int _z; 
      } t2{ 1, 2 };  // ok
      ​
      int T2::_z = 3;  // 注意:静态数据成员 _z 不能使用列表初始化进行初始化
    • 类中不能有 {} 和 = 直接初始化的非静态数据成员(即就地初始化)

      struct T3
      {int _x = 1;int _y{ 2 };
      } t3{ 0, 0 };  // error(C++11)

      注意:从 C++14 开始,也可以使用列表初始化来初始化类中使用 {} 和 = 初始化过的非静态数据成员

2.2.2 - 注意事项

聚合类型的定义并非递归的,即当一个类的非静态数据成员是非聚合类型时,这个类也可能是聚合类型

struct T1
{int _x;int _y;
private:int _z;
public:T1() : _x(1), _y(2), _z(3) { }
};
​
struct T2
{T1 _t1;double _d;
};
​
int main()
{T2 t2{ {}, 3.14 };return 0;
}

可以看到,T1 是非聚合类型,因为它有一个 private 的非静态数据成员,但 T2 依然是一个聚合类型,可以直接使用列表初始化来初始化对象 t2。

注意:使用列表初始化来初始化 t2 的非聚合类型成员 _t1 时,可以直接写一对空的大括号 {},这相当于调用 _t1 的默认构造函数

2.3 - initializer_list

2.3.1 - 基本使用

当编译器看到 { t1, t2, ..., tn } 时,便会生成一个 initializer_list<T> 类型的对象(其中 T 为元素的类型),它关联到一个 array<T, n>

#include <iostream>
using namespace std;
​
int main()
{auto il = { 10, 20, 30 };cout << typeid(il).name() << endl;  // class std::initializer_list<int>return 0;
}

对于聚合类型,编译器会将 array<T, n> 内的元素逐一分解并赋值给被初始化的对象,这相当于为该对象每个字段分别赋值

对于非聚合类型,如果该类存在一个接收 initializer_list<T> 类型的构造函数,则初始化时会将 initializer_list<T> 对象作为一个整体传给构造函数;如果该类不存在这样的构造函数,则 array<T, n> 内的元素会被编译器分解并传给相应的能接收这些参数的构造函数(比如列表中有 2 个元素,就传给带 2 个参数的构造函数,有 3 个元素,就传给带 3 个参数的构造函数,依次类推)

#include <iostream>
#include <vector>
using namespace std;
​
class Test
{
public:Test(int) { cout << "Test(int)" << endl; }
​Test(int, int) { cout << "Test(int, int)" << endl; }
};
​
int main()
{// vector (initializer_list<value_type> il, //      const allocator_type& alloc = allocator_type());vector<int> v{ 0, 1, 2, 3, 4 };for (const auto& e : v){cout << e << " ";}// 0 1 2 3 4cout << endl;Test t1{ 1 };  // Test(int)Test t2{ 1, 2 };  // Test(int, int)return 0;
}

2.3.2 - 源码剖析

#include <iostream>
​
template <class T>
class initializer_list
{
public:typedef T        value_type;typedef const T& reference;  // 说明对象永远为 const,不能被外部修改!typedef const T& const_reference;typedef size_t   size_type;typedef const T* iterator;  // 永远为 const 类型typedef const T* const_iterator;private:iterator  _M_array; // 用于存放用列表初始化中的元素size_type _M_len;   // 元素的个数
​// 注意:编译器可以调用 private 的构造函数!!!// 构造函数在调用之前,编译会先在外部准备好一个 array,// 同时把 array 的地址传入模板,并保存在 _M_array 中constexpr initializer_list(const_iterator __a, size_type __l):_M_array(__a), _M_len(__l) {};  // 注意该构造函数被放到 private 中!public:// 无参的构造函数constexpr initializer_list() : _M_array(0), _M_len(0) {} 
​// 用于获取元素的个数constexpr size_type size() const noexcept { return _M_len; }
​// 获取第一个元素的位置constexpr const_iterator begin() const noexcept { return _M_array; }
​// 获取最后一个元素的下一个位置constexpr const_iterator end() const noexcept{return begin() + _M_len;}
};

让模拟实现的 vector 也支持列表初始化

namespace yzz
{template<class T>class vector{public:typedef T* iterator;typedef const T* const_iterator;
​vector(std::initializer_list<T> il) :_start(new T[il.size()]),_finish(_start + il.size()),_end_of_storage(_finish){iterator v_it = _start;typename std::initializer_list<T>::iterator il_it = il.begin();while (il_it != il.end()){*v_it++ = *il_it++;}}// ... ...private:iterator _start;iterator _finish;iterator _end_of_storage;};
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/109734.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vm虚拟机克隆ubuntu

1. 使用vm虚拟机自带的克隆功能 2. 选择完整克隆&#xff0c;然后选择您克隆到哪里的目录 3. 点击编辑你克隆后的虚拟机&#xff0c;点网络适配器&#xff0c;然后点高级&#xff0c;点击生成mac地址&#xff08;由于唯一&#xff0c;所以需要重新生成&#xff09; 4. 开启虚拟…

【MySQL】分析SQL的几种方式

文章目录 一、查看SQL执行频率二、定位低效率执行SQL1. show processlist2. 慢查询日志 三、explain分析执行计划1. id2. select_type3. type4. key5. extra 四、show profile 一、查看SQL执行频率 show session status&#xff1a;显示 session 级的统计结果&#xff08;不写…

openGauss学习笔记-100 openGauss 数据库管理-管理数据库安全-客户端接入之用SSL进行安全的TCP/IP连接

文章目录 openGauss学习笔记-100 openGauss 数据库管理-管理数据库安全-客户端接入之用SSL进行安全的TCP/IP连接100.1 背景信息100.2 前提条件100.3 注意事项100.4 操作步骤100.5 相关参考 openGauss学习笔记-100 openGauss 数据库管理-管理数据库安全-客户端接入之用SSL进行安…

excel+requests管理测试用例接口自动化框架

背景&#xff1a; 某项目有多个接口&#xff0c;之前使用的unittest框架来管理测试用例&#xff0c;将每个接口的用例封装成一个py文件&#xff0c;接口有数据或者字段变动后&#xff0c;需要去每个py文件中找出变动的接口测试用例&#xff0c;维护起来不方便&#xff0c;为了…

双目视觉实战--单视图测量方法

目录 一.简介 二、2D变换 1. 等距变换&#xff08;欧式变换&#xff09; 2. 相似变换 3. 仿射变换 4. 射影变换&#xff08;透视变换&#xff09; 5. 结论 三、影消点与影消线 1. 平面上的线 2. 直线的交点 3. 2D无穷远点 4. 无穷远直线 5. 无穷远点的透视变换与仿…

Spring Cloud Gateway 使用 Redis 限流使用教程

从本文开始&#xff0c;笔者将总结 spring cloud 相关内容的教程 版本选择 为了适应 java8&#xff0c;笔者选择了下面的版本&#xff0c;后续会出 java17的以SpringBoot3.0.X为主的教程 SpringBoot 版本 2.6.5 SpringCloud 版本 2021.0.1 SpringCloudAlibaba 版本 2021.0.1.…

Prometheus的Pushgateway快速部署及使用

prometheus-pushgateway安装 一. Pushgateway简介 Pushgateway为Prometheus整体监控方案的功能组件之一&#xff0c;并做于一个独立的工具存在。它主要用于Prometheus无法直接拿到监控指标的场景&#xff0c;如监控源位于防火墙之后&#xff0c;Prometheus无法穿透防火墙&…

C++基础——内存分区模型

1 概述 C程序在执行是&#xff0c;将内存大致分为4个区域&#xff1a; 代码区&#xff1a;用于存放二进制代码&#xff0c;由操作系统进行管理全局区&#xff1a;存放全局变量和静态变量及常量栈区&#xff1a;由编译器自动分配释放&#xff0c;存放函数的参数、局部变量等堆…

回首往昔,初学编程那会写过的两段愚蠢代码

一、关于判断两个整数是否能整除的GW BASIC创意代码 记得上大学时第一个编程语言是BASIC&#xff0c;当时Visual Basic还没出世&#xff0c;QBASIC虽然已经在1991年随MS-DOS5.0推出了&#xff0c;但我们使用的还是 GW-BASIC&#xff0c; 使用的教材是谭浩强、田淑清编著的《BA…

【广州华锐互动】VR建筑安全培训体验为建筑行业人才培养提供有力支持

随着建筑行业的快速发展&#xff0c;建筑施工安全问题日益受到广泛关注。然而&#xff0c;传统的安全培训方式往往缺乏实践性和真实性&#xff0c;难以让员工真正掌握安全操作技能。近年来&#xff0c;虚拟现实(VR)技术的广泛应用为建筑施工安全培训提供了新的机遇。 虚拟现实技…

金融用户实践|分布式存储支持数据仓库业务系统性能验证

作者&#xff1a;深耕行业的 SmartX 金融团队 闫海涛 估值是指对资产或负债的价值进行评估的过程&#xff0c;这对于投资决策具有重要意义。每个金融公司资管业务人员都期望能够实现实时的业务估值&#xff0c;快速获取最新的数据和指标&#xff0c;从而做出更明智的投资决策。…

【数组的使用续篇】

文章目录 以数组的形式打印数组打印方法&#xff1a;Arrays.toString(数组名) 数组排序大小排序方法是 Arrays.sort(数组名) 创建一个自己的打印数组的方法自己创建一个冒泡排序两数之间交换方法 逆置数组打印核心思路还是 i 和 j 交换 总结 以数组的形式打印数组 打印方法&am…

软件测试用例设计方法-因果图法

边界值法是等价类划分法的补充&#xff0c;所以&#xff0c;它们是一对搭档。 那么&#xff0c;判定表法有没有它的搭档呢&#xff1f; 答案是&#xff0c;有的。那就是本篇文章分享的用例设计方法—— 因果图法 。 定义 因果图法&#xff1a; 用来处理等价类划分和边界值…

三维地图开发三维地图服务器

三维地图开发三维地图服务器 发布时间&#xff1a;2020-03-03 版权&#xff1a; 搭建离线地图服务主要是两个步骤&#xff1a;一是&#xff1a;下载离线地图服务需要的地图数据&#xff1b;二是&#xff1a;将下载的离线地图数据发布成地图服务&#xff1b;只有做好这两步&…

出差学小白知识No5:ubuntu连接开发板|上传源码包|板端运行的环境部署

1、ubuntu连接开发板&#xff1a; 在ubuntu终端通过ssh协议来连接开发板&#xff0c;例如&#xff1a; ssh root<IP_address> 即可 这篇文章中也有关于如何连接开发板的介绍&#xff0c;可以参考SOC侧跨域实现DDS通信总结 2、源码包上传 通过scp指令&#xff0c;在ub…

python使用dataset快速使用SQLite

目录 一、官网地址 二、安装 三、 快速使用 一、官网地址 GitHub - pudo/dataset: Easy-to-use data handling for SQL data stores with support for implicit table creation, bulk loading, and transactions. 二、安装 pip install dataset 如果是mysql&#xff0c;则…

堆/二叉堆详解[C/C++]

前言 堆是计算机科学中-类特殊的数据结构的统称。实现有很多,例如:大顶堆,小顶堆&#xff0c;斐波那契堆&#xff0c;左偏堆&#xff0c;斜堆等等。从子结点个数上可以分为二汊堆&#xff0c;N叉堆等等。本文将介绍的是二叉堆。 二叉堆的概念 1、引例 我们小时候&#xff0c;基…

left join时筛选条件对查询结果的

-- 创建表 CREATE TABLE table1 (id int(11) NOT NULL AUTO_INCREMENT,card_num varchar(60) DEFAULT NULL,customer_id varchar(60) DEFAULT NULL,PRIMARY KEY (id) ) ENGINE InnoDBAUTO_INCREMENT 12DEFAULT CHARSET utf8mb4 COMMENT 测试表1;-- 创建表 CREAT…

44springboot摄影跟拍预定管理系统

大家好✌&#xff01;我是CZ淡陌。一名专注以理论为基础实战为主的技术博主&#xff0c;将再这里为大家分享优质的实战项目&#xff0c;本人在Java毕业设计领域有多年的经验&#xff0c;陆续会更新更多优质的Java实战项目&#xff0c;希望你能有所收获&#xff0c;少走一些弯路…

basic_sr介绍

文章目录 pytorch基础知识和basicSR中用到的语法1.Sampler类与4种采样方式2.python dict的get方法使用3.prefetch_dataloader.py4. pytorch 并行和分布式训练4.1 选择要使用的cuda4.2 DataParallel使用方法常规使用方法保存和载入 4.3 DistributedDataParallel 5.wangdb 入门5.…