1. 使用列表初始化

在c++98/03中,对象的初始化方法有很多种,例如

int ar[3] = {1,2,3};
int arr[] = {1,2,3}; //普通数组
struct A{
int x;
struct B{
int y;
int z;
} b;
}a = {1, {3,4}}; //POD类型,可以直接使用memcpy复制的对象 int i = 0;
Foo foo = f;//拷贝初始化 int i(0);
Foo f(123); //直接初始化

在c++98/03中,普通数组合POD类型,可以通过初始化列表来进行初始化。c++11中,可以对任意类型对象使用初始化列表进行初始化。 
    列表初始化就是在一个变量后给出一个={},或者直接一个{},括号中含有需要的数据。

class Foo{
public:
Foo(int){};
private:
Foo(const Foo&);
};
int main(){
Foo a1(123);
Foo a2 = 123; //先隐式转换将123转化成一个Foo对象,然后调用operator=进行拷贝构造。
//错误,不能使用拷贝构造函数 Foo a3 = {123}; //列表初始化,虽然使用了=,但仍然是列表初始化,私有的拷贝构造无效
Foo a4{123}; //列表初始化
int a5 = {1};
int a6{2}; //new 操作符等可以使用圆括号进行初始化的地方,也可以使用列表初始化
int* a = new int{1, 2, 3};
double b = double{12.00};
int* arr = new int[3]{2,4,5}; //列表初始化还可以直接使用在函数的返回值上
struct Foo{
Foo(int, double){};
};
Foo func(void){
return {123, 12.00}; //返回构造的Foo对象,使用列表初始化
}
return 0;
}

2. 列表初始化的使用细节

c++11在进行列表初始化的时候,需要进行初始化的对象为聚合体。

c++聚合体

聚合类型的定义: 
(1)类型是一个普通数组 
(2)类型是一个类,而且: 
    (a)无用户自定义的构造函数 
    (b)无私有或保护的非静态数据成员 
    (c)无基类 
    (d)无虚函数 
    (e)不能有{}和=直接初始化的非静态数据成员

对于情形(e),举个例子
struct ST{
int x;
double y = 0.0;
};
ST s{1, 2.3}; //错误,不能有{}或=直接初始化的非静态数据成员

在c++98/03中,y这种非静态数据成员,本身就不能在声明时进行这种初始化工作,但是在c++11中,非静态数据成员也可以在声明的同时进行初始化工作(即使用{}或=初始化)。

struct ST{
int x;
int y;
ST(int a, int b):x(0), y(0){};
};
ST st{1, 2}; //这里进行初始时,不会进行列表初始化,而是调用的构造函数。因此,st.x = 0, st.y = 0.

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

struct ST{
int x;
double y;
private:
int z;
};
ST s{1, 2.4, 1};//错误,不是聚合类型,无法使用列表初始化
struct Foo{
ST st;
int x;
double y;
};
Foo foo{{}, 1, 2.3};//虽然ST并不是一个聚合类型,但是Foo为一个聚合类型。对于非聚合类型成员st初始化时,**使用空的{},相当于调用无参构造函数**

c++11的初始化列表赋值

对于一个聚合类型,使用初始化列表相当于对其中的每个元素分别赋值;而对于非集合类型,则需要先自定义一个合适的构造函数,此时使用初始化列表将调用它对应的构造函数。

3. 初始化列表

c++11中的stl容器拥有和未显示指定长度的数组一样的初始化能力,代码如下:

int arr[]{1,2,3};
std::map<std::string,int>mm={
{"123", 1}, {"hello", 2}, {"fuck", 3}
};
std::set<int> = {1,2,3};

stl中的容器是通过使用std::initializer_list 这个轻量级的模板来实现上述功能。 
std::initializer_list 
(1)它是一个轻量级的容器类型,内部定义了iterator等容器必须的概念 
(2)对于std::initializer_list< T>而言,它可以接受任意长度的初始化列表,但要求元素必须为类型T 
(3)它有三个成员接口: size(), begin(), end() 
(4)它只能被整体初始化或赋值,不能单独修改其中的某个元素(因为由上一条,知道它不提供接口) 
(5)它内部存放的是元素的引用,因此不会进行拷贝操作,所以需要注意在使用的时候保证内部存放的元素有效期

    std::initializer_list<int> list = {1,2,3,4};
size_t n = list.size(); // 4
list = {4, 5, 6}; //整体赋值

4. 防止类型收窄

类型收窄指的是导致数据内容发生变化或者精度丢失的隐式类型转换。类型收窄包含如下几种情况: 
(1)从一个浮点数隐式转换到一个整型数 
(2)从高精度浮点数转换为低精度浮点数 
(3)从一个整型数隐式转换为一个浮点数,并且超出了浮点数的表示范围,如 float x= (unsigned long long )-1; 
(4)从一个整型数转换为一个长度较短的整型数,并且超出了长度较短的整型数的表示范围。如char x = 3333; 
    在c++98/03中,出现上面的类型收窄的情况,编译器不会报错。在c++11中,可以通过列表初始化来检查以及防止类型收窄。

    int a = 1.1; //可以
int a = {1.1}; //出错

c++11——列表初始化的更多相关文章

  1. C++11 列表初始化

    在我们实际编程中,我们经常会碰到变量初始化的问题,对于不同的变量初始化的手段多种多样,比如说对于一个数组我们可以使用 int arr[] = {1,2,3}的方式初始化,又比如对于一个简单的结构体: ...

  2. C++11的初始化列表

      初始化是一个非常重要的语言特性,最常见的就是对对象进行初始化.在传统 C++ 中,不同的对象有着不同的初始化方法,例如普通数组.POD (plain old data,没有构造.析构和虚函数的类或 ...

  3. c++11之初始化列表

    一.前言     C++的学习中.我想每一个人都被变量定义和申明折磨过,比方我在大学笔试过的几家公司.都考察了const和变量,类型的不同排列组合,让你差别有啥不同.反正在学习C++过程中已经被折磨惯 ...

  4. C++11之列表初始化

    1. 在C++98中,标准允许使用花括号{}来对数组元素进行统一的集合(列表)初始化操作,如:int buf[] = {0};int arr[] = {1,2,3,4,5,6,7,8}; 可是对于自定 ...

  5. C++11常用特性介绍——列表初始化

    一.列表初始化 1)C++11以前,定义初始化的几种不同形式,如下: int data = 0;   //赋值初始化 int data = {0};   //花括号初始化 int data(0); / ...

  6. 第8课 列表初始化(3)_防止类型收窄、explicit关键字

    1. 防止类型收窄 (1)类型收窄:指的是导致数据内容发生变化或者精度丢失的隐式类型转换. (2)类型收窄的几种情况: ①从浮点数隐式转换为整型数,如int i=2.2; ②从高精度浮点数隐式转换为低 ...

  7. initializer_list、初始化列表、列表初始化

    什么是列表初始化 使用一个花括号来初始化变量,表现形式如下: std::vector<int>a{1,2,3,4,5}; 或者 std::vector<int>a = {1,2 ...

  8. 列表初始化 分析initializer_list<T>的实现

    列表初始化(1)_统一初始化 1. 统一初始化(Uniform Initialization) (1)在C++11之前,很多程序员特别是初学者对如何初始化一个变量或对象的问题很容易出现困惑.因为可以用 ...

  9. C++统一初始化语法(列表初始化)

    引言 要是世上不曾存在C++14和C++17该有多好!constexpr是好东西,但是让编译器开发者痛不欲生:新标准库的确好用,但改语法细节未必是明智之举,尤其是3年一次的频繁改动.C++带了太多历史 ...

随机推荐

  1. Cookie application session

    •Application 对象是存储于服务器的全局变量 •Cookie 存储信息于客户端 •Session 对象用于在服务器端存储用户的信息,在用户结束会话时被清除 1.将信息写入Cookies 中/ ...

  2. Java-ThreadLocal,Java中特殊的线程绑定机制

    在DRP项目中,我们使用了ThreadLocal来创建Connection连接,避免了一直以参数的形式将Connection向下传递(传递connection的目的是由于jdbc事务要求确保使用同一个 ...

  3. 自然语言交流系统 phxnet团队 创新实训 个人博客 (一)

    项目涉及链表操作,遂整理: 使用链表结构可以克服数组链表需要预先知道数据大小的缺点,链表结构可以充分利用计算机内存空间,实现灵活的内存动态管理.但是链表失去了数组随机读取的优点,同时链表由于增加了结点 ...

  4. double类型保留2位小数

    double d = 12.2289; java.text.DecimalFormat df = new java.text.DecimalFormat("#.00"); Syst ...

  5. C 字符串操作函数

    针对C风格的字符串(char p[n];): 长度(strlen).追加(strcat, strncat).比较(strcmp, strncmp).查找(strchr, strstr)等. --带n的 ...

  6. 用C语言实现解析简单配置文件的小工具

    本文介绍作者写的一个小工具,简单的代码中包含了C语言对字符串的处理技巧,对文本文件的简单解析,二进制文件的数据复制的方法,以及格式化输出文本文件的示例. 工具的输入是如下内容的配置文件: ;资源管理器 ...

  7. linux -- chown修改文件拥有者和所在组

    chown将指定文件的拥有者改为指定的用户或组,用户可以是用户名或者用户ID:组可以是组名或者组ID:文件是以空格分开的要改变权限的文件列表,支持通配符.系统管理员经常使用chown命令,在将文件拷贝 ...

  8. 经常使用的CSS Hack技术集锦

    来源:http://www.ido321.com/938.html 一.什么是CSS Hack? 不同的浏览器对CSS的解析结果是不同的,因此会导致同样的CSS输出的页面效果不同,这就须要CSS Ha ...

  9. 五大移动GPU厂商

    <谁能笑傲江湖?移动处理器门派那些事儿>一文中我们把2012年的移动处理器的厂商做了一番介绍,并依照各自的属性给划分了门派.既然把他们称为江湖门派.那么每一个门派总要有自己的绝活.移动处理 ...

  10. arugsJS 入门

    一款优秀的前端框架——AngularJS     前  言 AngularJS是一款为了克服HTML在构建应用上的不足而设计的优秀的前端JS框架.AngularJS有着诸多特性,最为核心的是:MVC. ...