C++类和对象之初始化列表的使用方式

 更新时间:2025年05月12日 09:57:45   作者:zzh_zao  
这篇文章主要介绍了C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

C++初始化列表详解:性能优化与正确实践

在C++编程中,初始化列表是构造函数的重要组成部分,它不仅能提升代码性能,还能确保成员变量被正确初始化。本文将深入探讨初始化列表的语法、应用场景及最佳实践。

什么是初始化列表?

初始化列表是构造函数的一部分,用于在对象创建时直接初始化成员变量。

它位于构造函数参数列表之后,函数体之前,使用冒号(:)和逗号(,)分隔各成员的初始化操作。

class MyClass {
private:
    int value;
    std::string name;
    const double pi;
    int& ref;

public:
    // 使用初始化列表的构造函数
    MyClass(int val, const std::string& nm, int& r)
        : value(val), name(nm), pi(3.14159), ref(r) {
        // 构造函数体
    }
};

初始化列表的三大核心作用

1. 性能优化:避免不必要的赋值操作

对于非基本类型(如std::stringstd::vector),初始化列表可以直接调用其构造函数,而不是先默认构造再赋值。

对比示例:

// 使用初始化列表(高效)
class A {
public:
    A(const std::string& str) : data(str) {} // 直接构造
private:
    std::string data;
};

// 使用赋值(低效)
class B {
public:
    B(const std::string& str) { data = str; } // 默认构造 + 赋值
private:
    std::string data;
};

2. 强制初始化:处理const和引用成员

const成员和引用必须在初始化时赋值,初始化列表是唯一的方式。

class Config {
private:
    const int maxSize;      // 常量成员
    std::string& filePath;  // 引用成员

public:
    Config(int size, std::string& path) 
        : maxSize(size), filePath(path) {} // 必须在初始化列表中赋值
};

3. 基类初始化:正确调用父类构造函数

当基类没有默认构造函数时,必须通过初始化列表显式调用其带参构造函数。

class Base {
public:
    Base(int value) { /* ... */ }
};

class Derived : public Base {
public:
    Derived(int x) : Base(x) { /* ... */ } // 调用基类构造函数
};

4.必须使用初始化列表的情况

在C++中,引用、const成员变量以及没有默认构造函数的类类型成员变量必须在构造函数初始化列表中进行初始化,这是由它们的语义和C++对象生命周期的规则决定的。

1. 引用(Reference)

引用的本质是对象的别名,一旦绑定到某个对象就无法重新绑定。因此:

  • 必须在创建时初始化:引用没有“未初始化”状态,必须在定义时指定其引用的对象。
  • 构造函数体执行前成员已初始化:构造函数体中的代码执行时,成员变量已经完成初始化。若在构造函数体内对引用赋值,实际上是对已初始化的引用进行赋值操作(改变被引用对象的值),而非初始化引用本身。

示例:

class Example {
private:
    int& ref;  // 引用必须初始化
public:
    Example(int& value) : ref(value) {}  // 正确:初始化列表中初始化
};

2. const 成员变量

const成员变量的值在对象的生命周期内不可修改,因此:

  • 必须在初始化时赋值const变量一旦初始化就不能再被赋值。
  • 初始化列表是唯一机会:构造函数体执行前,const成员必须已经被赋予初始值。

示例:

class Example {
private:
    const int value;  // const成员必须初始化
public:
    Example(int val) : value(val) {}  // 正确:初始化列表中初始化
};

3. 没有默认构造函数的类类型成员

如果一个类没有默认构造函数(即没有无参构造函数),则在创建该类的对象时必须显式提供参数。因此:

  • 必须通过参数初始化:编译器无法默认构造该成员,必须显式调用其带参构造函数。
  • 初始化列表提供了显式调用的机会:在初始化列表中,可以指定参数来调用成员的带参构造函数。

示例:

class Inner {
public:
    Inner(int x) {}  // 只有带参构造函数
};

class Example {
private:
    Inner inner;  // Inner没有默认构造函数
public:
    Example(int x) : inner(x) {}  // 正确:显式调用Inner的带参构造函数
};

为什么不能在构造函数体中初始化?

构造函数体中的代码执行时,成员变量已经完成初始化(默认初始化或编译器生成的初始化)。因此:

  • 引用和const成员:无法在构造函数体中重新初始化,因为它们必须在初始化时就确定值。
  • 无默认构造函数的类成员:如果未在初始化列表中显式构造,编译器会尝试调用其默认构造函数,但由于该类没有默认构造函数,会导致编译错误。

初始化列表的作用是在对象的内存分配后、构造函数体执行前,显式控制成员变量的初始化过程。对于引用、const成员和无默认构造函数的类成员,初始化列表是唯一能满足其初始化语义的地方。若不使用初始化列表,代码将因“未初始化的引用/const成员”或“无法默认构造的类成员”而编译失败。

初始化列表的语法规则

1. 初始化顺序由声明顺序决定

成员变量的初始化顺序由其在类中声明的顺序决定,而非初始化列表中的顺序。错误的顺序可能导致未定义行为。

class Example {
private:
    int a;
    int b;

public:
    // 危险:初始化列表顺序与声明顺序不一致
    Example(int value) : b(value), a(b) {} // a先被初始化,但此时b未初始化
};

2. 支持表达式初始化

可以使用常量、函数返回值或其他成员变量进行初始化。

class Point {
private:
    int x;
    int y;
    int distance;

public:
    Point(int _x, int _y) 
        : x(_x), y(_y), 
          distance(calculateDistance(_x, _y)) {} // 使用函数返回值初始化

    int calculateDistance(int x, int y) const {
        return std::sqrt(x*x + y*y);
    }
};

初始化列表 vs 构造函数体赋值

特性初始化列表构造函数体赋值
执行时机对象创建时对象创建后
性能通常更高效可能涉及额外的赋值操作
const/引用成员支持不支持
基类初始化必须使用不可用

总结

初始化列表总结:

  • 无论是否显⽰写初始化列表,每个构造函数都有初始化列表;
  • 无论是否在初始化列表显⽰初始化成员变量,每个成员变量都要⾛初始化列表初始化;

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 用C语言实现井字棋游戏代码

    用C语言实现井字棋游戏代码

    大家好,本篇文章主要讲的是用C语言实现井字棋游戏代码,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2022-01-01
  • C++实现日期类的方法详解

    C++实现日期类的方法详解

    这篇文章主要给大家介绍了C++实现日期类的方法,文中通过代码示例给大家介绍的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下
    2024-01-01
  • C语言编程递归算法实现汉诺塔

    C语言编程递归算法实现汉诺塔

    递归,大家都了解,著名的斐波那契数,就为该知识点的经典例题。今天来看看更为经典的递归题汉诺塔不过这其实是数学问题,先来看看汉诺塔
    2021-09-09
  • C++基于easyx图形库实现推箱子游戏

    C++基于easyx图形库实现推箱子游戏

    这篇文章主要为大家详细介绍了C++基于easyx图形库实现推箱子游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-06-06
  • c++如何实现归并两个有序链表

    c++如何实现归并两个有序链表

    这篇文章主要介绍了c++如何实现归并两个有序链表,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-07-07
  • c++数组排序的5种方法实例代码

    c++数组排序的5种方法实例代码

    大家还在为大小排序而烦恼吗?今天让我我大家解决这个问题吧,下面这篇文章主要给大家介绍了关于c++数组排序的5种方法,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-01-01
  • C++14新特性的所有知识点全在这

    C++14新特性的所有知识点全在这

    这篇文章主要介绍了C++14新特性的所有知识点全在这,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-07-07
  • Qt Creator使用教程的简单说明

    Qt Creator使用教程的简单说明

    如今 Qt Creator 功能十分强大了,包含项目模板生成、代码编辑、UI 设计、QML 界面编辑、调试程序、上下文帮助等丰富功能,本文就详细的介绍一下如何使用
    2021-08-08
  • C++ 函数重载之规则、实现与实战案例解析

    C++ 函数重载之规则、实现与实战案例解析

    本文主要介绍了C++函数重载的核心规则,包括函数重载的匹配原则、与默认参数的冲突处理以及实战场景中的重载应用,通过实例展示了如何实现一个通用计算器,支持整数和浮点数的加减乘除运算,并通过参数校验避免除零错误,感兴趣的朋友跟随小编一起看看吧
    2026-02-02
  • c语言通过opencv实现轮廓处理与切割

    c语言通过opencv实现轮廓处理与切割

    这篇文章主要介绍了c语言通过opencv实现轮廓处理与切割,具有一定借鉴价值,需要的朋友可以参考下
    2018-01-01

最新评论