C++一分钟之-C++11新特性:初始化列表

本文涉及的产品
实时计算 Flink 版,5000CU*H 3个月
实时数仓Hologres,5000CU*H 100GB 3个月
智能开放搜索 OpenSearch行业算法版,1GB 20LCU 1个月
简介: 【6月更文挑战第21天】C++11的初始化列表增强语言表现力,简化对象构造,特别是在处理容器和数组时。它允许直接初始化成员变量,提升代码清晰度和性能。使用时要注意无默认构造函数可能导致编译错误,成员初始化顺序应与声明顺序一致,且在重载构造函数时避免歧义。利用编译器警告能帮助避免陷阱。初始化列表是高效编程的关键,但需谨慎使用。

C++11引入了一系列改进,极大地增强了语言的表达力和效率,其中初始化列表(Initializer Lists)是一个尤为重要的新特性。它提供了一种更为直观和高效的构造复杂对象的方式,尤其是在处理容器、数组和其他聚合类型时。本文将深入浅出地探讨初始化列表的使用、常见问题、易错点以及如何避免这些陷阱,并通过代码示例加以说明。
image.png

初始化列表基础

初始化列表允许在创建对象时直接初始化其成员变量,替代了传统的构造函数体内赋值。这不仅提升了代码的清晰度,还避免了不必要的默认构造-赋值过程,提高了性能。

class Point {
   
   
public:
    int x, y;
    Point(int x, int y) : x(x), y(y) {
   
   } // 使用初始化列表
};

Point p(10, 20); // 直接通过参数列表初始化

使用场景

对象与数组

对于内置类型数组和类的对象数组,初始化列表提供了一种简洁的初始化方式。

int arr[] = {
   
   1, 2, 3, 4, 5}; // 数组初始化
Point points[] = {
   
   {
   
   1, 2}, {
   
   3, 4}}; // 对象数组初始化

标准库容器

初始化列表特别适用于STL容器的初始化,如std::vectorstd::map等。

std::vector<int> vec = {
   
   1, 2, 3, 4};
std::map<std::string, int> map = {
   
   {
   
   "apple", 1}, {
   
   "banana", 2}};

常见问题与易错点

默认构造函数的省略

当类没有默认构造函数时,直接使用花括号初始化可能引发编译错误。

class NoDefaultConstructor {
   
   
public:
    NoDefaultConstructor(int x) : x_(x) {
   
   }
private:
    int x_;
};

NoDefaultConstructor ndc{
   
   }; // 错误!没有默认构造函数

初始化顺序与成员声明顺序

成员变量的初始化顺序严格遵循它们在类声明中的顺序,而不是初始化列表中的顺序。

class MyClass {
   
   
public:
    MyClass(int a, int b) : b(b), a(a) {
   
   } // 注意:b先于a被初始化
private:
    int a, b;
};

初始化列表与构造函数重载

在有多个构造函数重载的情况下,编译器可能无法确定使用哪个构造函数,尤其是当初始化列表提供的信息不足以区分时。

如何避免易错点

明确构造函数意图

确保每个构造函数都有清晰的职责划分,必要时通过提供默认参数或使用 delegating constructors(委托构造函数)来避免歧义。

MyClass(int a = 0, int b = 0) : a(a), b(b) {
   
   } // 添加默认参数

注意成员声明顺序

在设计类时,应考虑成员变量的初始化顺序,尽量避免依赖于特定初始化顺序的逻辑。

利用编译器警告和错误

现代编译器提供了丰富的警告选项,如-Wreorder(GCC)可以帮助发现成员初始化顺序与声明顺序不一致的问题。

结语

初始化列表是C++11中的一项强大特性,它简化了对象的初始化过程,提升了代码的可读性和执行效率。正确理解和应用这一特性,能够使你的C++编程之旅更加顺畅。然而,正如所有强大的工具一样,初始化列表也需谨慎使用,避免陷入常见的陷阱之中。通过本文的介绍和示例,希望能帮助你更好地掌握初始化列表的精髓,编写出更加高效、优雅的C++代码。

目录
相关文章
|
4月前
|
编译器 程序员 定位技术
C++ 20新特性之Concepts
在C++ 20之前,我们在编写泛型代码时,模板参数的约束往往通过复杂的SFINAE(Substitution Failure Is Not An Error)策略或繁琐的Traits类来实现。这不仅难以阅读,也非常容易出错,导致很多程序员在提及泛型编程时,总是心有余悸、脊背发凉。 在没有引入Concepts之前,我们只能依靠经验和技巧来解读编译器给出的错误信息,很容易陷入“类型迷路”。这就好比在没有GPS导航的年代,我们依靠复杂的地图和模糊的方向指示去一个陌生的地点,很容易迷路。而Concepts的引入,就像是给C++的模板系统安装了一个GPS导航仪
158 59
|
3月前
|
安全 编译器 C++
【C++11】新特性
`C++11`是2011年发布的`C++`重要版本,引入了约140个新特性和600个缺陷修复。其中,列表初始化(List Initialization)提供了一种更统一、更灵活和更安全的初始化方式,支持内置类型和满足特定条件的自定义类型。此外,`C++11`还引入了`auto`关键字用于自动类型推导,简化了复杂类型的声明,提高了代码的可读性和可维护性。`decltype`则用于根据表达式推导类型,增强了编译时类型检查的能力,特别适用于模板和泛型编程。
29 2
|
4月前
|
存储 编译器 数据安全/隐私保护
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解2
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
63 3
|
4月前
|
编译器 C++
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解1
【C++篇】C++类与对象深度解析(四):初始化列表、类型转换与static成员详解
68 3
|
4月前
|
C++
C++ 20新特性之结构化绑定
在C++ 20出现之前,当我们需要访问一个结构体或类的多个成员时,通常使用.或->操作符。对于复杂的数据结构,这种访问方式往往会显得冗长,也难以理解。C++ 20中引入的结构化绑定允许我们直接从一个聚合类型(比如:tuple、struct、class等)中提取出多个成员,并为它们分别命名。这一特性大大简化了对复杂数据结构的访问方式,使代码更加清晰、易读。
59 0
|
4月前
|
C++
C++构造函数初始化类对象
C++构造函数初始化类对象
29 0
|
23天前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
63 19
|
23天前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
42 13
|
23天前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
46 5
|
23天前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
36 5

热门文章

最新文章