IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> C++初始化列表 -> 正文阅读

[C++知识库]C++初始化列表

我们知道创建对象时,编译器通过调用构造函数来给成员变量赋值。通过调试可以发现,进入构造函数体内时,成员变量已经创建好并存储了随机值,且赋值可以多次但初始化只有一次,所以构造函数体内的语句是赋值而不是初始化。
这说明了,在进行赋值操作前,系统已经完成了对成员变量的初始化。
接下来,看一个例子:

class Test1 {
    public:
    //无参构造函数
    Test1() {
        cout << "Construct Test1" << endl;
    }
    //带参构造函数
    Test1(int a) {
        this->a = a;
        cout << "Construct Test1 with parameter" << endl;
    }
    //拷贝构造函数
    Test1(const Test1& t1) {
        cout << "Copy constructor for Test1" << endl;
        this->a = t1.a;
    }
    //赋值运算符
    Test1& operator= (const Test1& t1) {
        cout << "Assignment for Test1" << endl;
        this->a = t1.a;
        return *this;
    }
    int a;
};

class Test2 {
    public:
    Test1 test1;
    Test2(Test1 &t1) {
        test1 = t1;
    }
};

int main(int argc, const char * argv[]) {
    
    Test1 t1;
    Test2 t2(t1);
    
    return 0;
}

运行程序,输出为:

Construct Test1
Construct Test1
assignment for Test1
  • 第一行为Test1 t1;创建时调用的构建函数。我们接下来打印一下初始化出来的t1里的a值:
cout<< t1.a <<endl;

输出:

1966080001

输出一个随机值。这是因为在无参构造函数中没有默认初始化赋值。
前面输出结果

  • 第二行也是一个Construct Test1,这个就是前面提到的,先为t2成员变量test1构建时调用的默认构造函数。
  • 第三行输出对应Test2的赋值运算符,对test1执行赋值操作。

定义

初始化列表以冒号开头,后跟一系列以逗号分隔的初始化字段。
例:

class Date {
public:
	Date(int year = 1, int month = 1, int day = 1)
		: _year(year)
		, _month(month)
		, _day(day)
	{}
private:
	int _year; // 声明变量
	int _month;
	int _day;
};
class Test2 {
    public:
    Test1 test1;
    Test2(Test1 &t1) : test1(t1) {
//        test1 = t1;
    }
};

对test2例修改,继续执行

Test1 t1;
Test2 t2(t1);

发现结果变成了:

Construct Test1
Copy constructor for Test1
  • 第一行输出对应 调用代码的第一行
  • 第二行输出对应Test2的初始化列表,直接调用拷贝构造函数初始化test1,省去了调用默认构造函数的过程。

初始化类的成员有两种方式,一是使用初始化列表,二是在构造函数体内进行赋值操作。
主要是性能问题,对于内置类型,如intfloat等,使用初始化类表和在构造函数体内初始化差别不是很大,但是对于类类型来说,最好使用初始化列表,为什么呢?由测试可知,使用初始化列表少了一次调用默认构造函数的过程,这对于数据密集型的类来说,是非常高效的。

特点

  • 初始化列表只能用在构造函数中。
  • 变量只能初始化一次,故每个成员变量只能在初始化列表出现一次。
  • 三种成员变量必须放在初始化列表进行初始化:
    1. 引用成员变量
    2. const 常变量成员
    3. 无默认构造函数的自定义类型成员。

对于以下代码:

class Test3 {
    public:
    Test3(int a):i(a){}
    int i;
};
class Test4 {
    public:
    Test3 test3 ;
    Test4(Test3 &t3) {
        test3 = t1;
    }
};

无法通过编译,编译器提示:Constructor for 'Test4' must explicitly initialize the member 'test3' which does not have a default constructor,属于无默认构造函数的自定义类型成员
正确如下:

class Test4 {
    public:
    Test3 test3 ;
    Test4(int x) : test3(x) {
//        test3 = t1;
    }
};

此外,成员变量在类中的声明次序就是其在初始化列表中的初始化顺序,初始化顺序与初始化列表顺序无关。

class A {
    public:
    A() :_a1(1) , _a2(_a1) {}
    void Print() {
        cout << _a1 << " " << _a2 << endl;
    }
    private:
    int _a2;
    int _a1;
};

调用:

A().Print();

输出:

1 386695169

类中声明成员变量的次序是先 a2a1,所以初始化时也是这个顺序,就导致 a2 被初始化为随机值。实际开发中,最好按照相同的顺序初始化。

  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-03-13 21:33:09  更:2022-03-13 21:34:29 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 4:48:22-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码