0

0

C++组合类型中默认成员初始化方法

P粉602998670

P粉602998670

发布时间:2025-09-11 11:04:01

|

838人浏览过

|

来源于php中文网

原创

C++组合类型成员的默认初始化行为取决于成员类型、类内初始化器(ICMI)和构造函数定义。基本类型成员在局部对象中若无ICMI或构造函数初始化,则为未定义值(垃圾值);全局或静态对象及用{}初始化时会被零初始化。类类型成员优先使用ICMI,否则调用其默认构造函数,若不存在则编译失败。ICMI提供保底初始化,但构造函数初始化列表优先级更高。为避免垃圾值,应优先使用ICMI、构造函数初始化列表,并避免在构造函数体内赋值基本类型成员。

c++组合类型中默认成员初始化方法

在C++组合类型中,成员的默认初始化行为并非一成不变,它复杂地取决于成员的类型(是基本类型还是类类型)、它们是否拥有类内初始化器(ICMI),以及包含它们的类是否定义了构造函数。简单来说,基本类型成员在大多数局部语境下默认是未初始化的,而类类型成员则会尝试调用其默认构造函数,如果存在的话。C++11引入的类内成员初始化器则为所有成员提供了一个强大的“保底”初始化机制。

解决方案

理解C++组合类型(如

struct
class
)中成员的默认初始化,我们首先要区分几种关键情况。这不仅仅是语法问题,更是关乎程序正确性与效率的核心。

1. 基本类型成员(如

int
,
double
, 指针等)

当一个组合类型对象被默认初始化(例如,

MyStruct obj;
)时:

立即学习C++免费学习笔记(深入)”;

  • 作为局部对象成员时: 如果这些基本类型成员没有在类内定义时提供初始化器(即C++11后的ICMI),也没有在构造函数的成员初始化列表中显式初始化,那么它们的值是未定义的(uninitialized)。这意味着它们可能含有任何“垃圾”值,使用它们会导致未定义行为。
  • 作为全局或静态存储期对象成员时: 它们会被零初始化。这是C++标准对这类变量的特殊规定,以确保程序启动时的确定性。
  • 使用
    {}
    进行值初始化时:
    例如
    MyStruct obj{};
    new MyStruct{};
    ,即使是基本类型成员,也会被零初始化。这是一种安全地初始化基本类型成员的推荐方式。

2. 类类型成员(自定义类、

std::string
,
std::vector
等)

  • 拥有类内成员初始化器(ICMI)时: 如果成员在类定义时就提供了初始化器(例如
    std::string s = "hello";
    std::vector v{1, 2, 3};
    ),那么这个ICMI会优先生效。
  • 没有ICMI,但类型拥有可访问的默认构造函数时: 编译器会调用该成员类型的默认构造函数进行初始化。如果这个默认构造函数是编译器隐式生成的,它会递归地对成员的成员进行初始化。
  • 没有ICMI,且类型没有可访问的默认构造函数时: 编译会失败。这是因为编译器无法知道如何构造这个成员。在这种情况下,你必须在包含它的类的构造函数初始化列表中显式地初始化这个成员。

3. 类内成员初始化器 (In-class Member Initializers, ICMIs) (C++11及以后)

这是C++11引入的一项重要特性,它允许你在类定义中直接为非静态数据成员提供默认值。

  • 优先级: ICMI的优先级低于构造函数的成员初始化列表。如果构造函数初始化列表显式初始化了某个成员,那么ICMI会被忽略。否则,ICMI就会生效。
  • 作用: ICMI为成员提供了一个“保底”的默认值,极大地简化了构造函数的编写,并减少了因遗漏初始化而导致的未定义行为。

代码示例:

#include 
#include 
#include 

struct MyStruct {
    int i;                  // 基本类型,无ICMI。局部对象时未初始化。
    double d = 3.14;        // 基本类型,有ICMI。
    std::string s;          // 类类型,无ICMI。调用std::string的默认构造函数。
    std::vector v{1, 2, 3}; // 类类型,有ICMI。
    int arr[2];             // 基本类型数组,无ICMI。局部对象时元素未初始化。

    // 默认构造函数,没有显式初始化 i 和 arr
    MyStruct() {
        std::cout << "MyStruct default constructor called." << std::endl;
        // 注意:i 和 arr 此时仍是未定义的,除非在这里或初始化列表中显式赋值。
        // d, s, v 已经通过ICMI或其默认构造函数完成初始化。
    }

    // 带参数的构造函数,显式初始化了 i
    MyStruct(int val) : i(val) {
        std::cout << "MyStruct parameterized constructor called." << std::endl;
        // d, s, v 依然会通过ICMI或其默认构造函数初始化。
    }
};

struct AnotherStruct {
    int x = 10; // ICMI
    AnotherStruct() = default; // 编译器生成的默认构造函数会使用ICMI
};

struct YetAnotherStruct {
    int y;
    YetAnotherStruct() : y(20) {} // 构造函数初始化列表显式初始化 y
};

struct NoDefaultCtorMember {
    NoDefaultCtorMember(int val) : value(val) {} // 只有带参数的构造函数
    int value;
};

struct ContainsNoDefaultCtorMember {
    // NoDefaultCtorMember member_obj; // 编译错误:NoDefaultCtorMember没有可访问的默认构造函数
    NoDefaultCtorMember member_obj{5}; // OK,通过ICMI提供初始化参数
    ContainsNoDefaultCtorMember() {}
};

int main() {
    std::cout << "--- 默认初始化 MyStruct ms; ---" << std::endl;
    MyStruct ms; // 局部对象,i 和 arr 是未定义的
    std::cout << "ms.i: " << ms.i << std::endl; // 输出垃圾值
    std::cout << "ms.d: " << ms.d << std::endl; // 输出3.14
    std::cout << "ms.s: '" << ms.s << "'" << std::endl; // 输出空字符串
    std::cout << "ms.v size: " << ms.v.size() << std::endl; // 输出3
    std::cout << "ms.arr[0]: " << ms.arr[0] << std::endl; // 输出垃圾值

    std::cout << "\n--- 值初始化 MyStruct ms_value{}; ---" << std::endl;
    MyStruct ms_value{}; // 所有基本类型成员会被零初始化
    std::cout << "ms_value.i: " << ms_value.i << std::endl; // 输出0
    std::cout << "ms_value.d: " << ms_value.d << std::endl; // 输出3.14 (ICMI优先)
    std::cout << "ms_value.s: '" << ms_value.s << "'" << std::endl; // 输出空字符串
    std::cout << "ms_value.v size: " << ms_value.v.size() << std::endl; // 输出3
    std::cout << "ms_value.arr[0]: " << ms_value.arr[0] << std::endl; // 输出0

    std::cout << "\n--- MyStruct parameterized constructor ---" << std::endl;
    MyStruct ms_param(100);
    std::cout << "ms_param.i: " << ms_param.i << std::endl; // 输出100

    std::cout << "\n--- AnotherStruct as; ---" << std::endl;
    AnotherStruct as;
    std::cout << "as.x: " << as.x << std::endl; // 输出10

    std::cout << "\n--- YetAnotherStruct yas; ---" << std::endl;
    YetAnotherStruct yas;
    std::cout << "yas.y: " << yas.y << std::endl; // 输出20

    std::cout << "\n--- ContainsNoDefaultCtorMember cndcm; ---" << std::endl;
    ContainsNoDefaultCtorMember cndcm;
    std::cout << "cndcm.member_obj.value: " << cndcm.member_obj.value << std::endl; // 输出5

    return 0;
}

C++类成员变量的默认初始化,哪些情况会导致“垃圾值”?

在我看来,C++在效率与安全性之间做了权衡。对于基本类型(

int
,
float
, 指针等),C++默认情况下并不会强制进行零初始化,这主要是为了避免在不需要时产生额外的性能开销。这种设计哲学导致了“垃圾值”的出现,如果开发者不注意,很容易踩坑。

LALAL.AI
LALAL.AI

AI人声去除器和声乐提取工具

下载

具体来说,以下几种情况最容易导致成员变量含有“垃圾值”:

  1. 局部组合类型对象的基本类型成员未被显式初始化: 这是最常见的陷阱。当你声明一个局部对象,例如

    MyStruct ms;
    ,如果
    MyStruct
    中的
    int
    double
    成员没有在类定义中提供ICMI,也没有在
    MyStruct
    的构造函数(无论是默认构造函数还是用户定义的其他构造函数)的成员初始化列表中初始化,更没有在构造函数体内部赋值,那么这些成员的值将是未定义的。它们会保留创建时内存区域中的任意二进制数据,这就是我们常说的“垃圾值”。

    struct BadExample {
        int value; // 没有ICMI
        // BadExample() {} // 默认构造函数没有初始化 value
    };
    
    int main() {
        BadExample be;
        std::cout << be.value << std::endl; // 极可能输出垃圾值
    }
  2. 基本类型数组作为成员,且未被显式初始化: 数组的初始化规则与单个基本类型成员类似。如果一个

    int arr[10];
    作为类成员,且没有ICMI或构造函数初始化,那么
    arr
    中的每个元素都将是未定义的。

    struct AnotherBadExample {
        int data[5]; // 没有ICMI
        // AnotherBadExample() {}
    };
    
    int main() {
        AnotherBadExample abe;
        std::cout << abe.data[0] << std::endl; // 极可能输出垃圾值
    }
  3. 构造函数体内部赋值而非初始化列表: 这是一个微妙但重要的点。如果你在构造函数体内部对基本类型成员进行赋值,例如:

    struct MyClass {
        int x;
        MyClass() {
            x = 10; // 在这里赋值
        }
    };

    x = 10;
    执行之前,
    x
    已经经历了默认初始化阶段。对于基本类型,这意味着它首先是未定义的。虽然最终它会被赋值为10,但在赋值前的一小段时间内,它确实是未定义的。虽然这通常不会导致运行时错误,但从严格的C++语义上讲,它在构造函数体执行前确实“拥有”了垃圾值。更重要的是,对于非基本类型,这会导致先默认构造,再调用赋值运算符,效率较低且可能产生不必要的副作用。

与此形成对比的是,全局或静态存储期的对象,其所有成员(包括基本类型)都会被零初始化。而使用

{}
进行值初始化时,也能确保所有基本类型成员被零初始化,从而避免垃圾值。所以,理解这些细微的差别,对编写健壮的C++代码至关重要。

如何确保C++组合类型成员总是被正确初始化?

确保C++组合类型成员总是被正确初始化,是编写健壮、可维护代码的关键一步。这不仅能避免未定义行为,还能提升代码的清晰度。在我多年的开发经验中,我总结出几种行之有效的方法,它们各有侧重,但目标一致:

  1. 优先使用类内成员初始化器(ICMI) (C++11及以后) 这是我个人最推荐的方式,因为它简洁、直观,并且提供了一个“保底”的默认值。当一个新成员被添加到类中时,只需在声明处提供ICMI,就能确保所有构造函数在没有显式初始化该成员时,它也能获得一个合理的值。这大大减少了遗漏初始化的可能性。

    class User {
    public:
        std::string name = "Guest"; // ICMI for string
        int id = 0;                 // ICMI for int
        bool active = true;         // ICMI for bool
    
        User() = default; // 默认构造函数会使用ICMI
        User(const std::string& n, int i) : name(n), id(i) {} // 构造函数初始化列表优先
    };
    
    // User u; // name="Guest", id=0, active=true
    // User u2("Alice", 123); // name="Alice", id=123, active=true

    ICMI的优点在于其“就近原则”,成员的默认值与其声明紧密结合,易于理解和维护。

  2. 善用构造函数成员初始化列表 对于那些需要根据构造函数参数来初始化的成员,或者需要执行更复杂初始化逻辑的成员,构造函数成员初始化列表是最佳选择。它的优势在于:

    • 效率: 成员在构造函数体执行之前就已经被初始化了,避免了先默认构造再赋值的二次操作,特别是对于类类型成员,这可以避免不必要的临时对象和赋值操作。
    • 强制性: 对于没有默认构造函数的类类型成员,你必须在初始化列表中对其进行初始化,否则编译会失败。
    • 常量和引用成员:
      const
      成员和引用成员只能在初始化列表中初始化,因为它们一旦创建就不能被赋值。
      class Point {
      const int x; // const 成员
      int& y_ref;  // 引用成员
      std::string label;

    public: Point(int val_x, int& val_y, const std::string& l) : x(val_x), y_ref(val_y), label(l) { // 必须使用初始化列表 // 构造函数体内部不能再对 x 和 y_ref 赋值 } };

  3. 避免在构造函数体内部对成员进行赋值 如前所述,在构造函数体内部对成员赋值,意味着该成员首先经历了默认初始化(对于基本类型是未定义,

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

483

2023.08.02

css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

580

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

102

2025.10.23

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1503

2023.10.24

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1503

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

233

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

87

2025.10.17

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

531

2023.09.20

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

14

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 8万人学习

C 教程
C 教程

共75课时 | 4.3万人学习

C++教程
C++教程

共115课时 | 14.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号