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

2025-05-13 01:50

本文主要是介绍C++类和对象之初始化列表的使用方式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《C++类和对象之初始化列表的使用方式》:本文主要介绍C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教...

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

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

什么是初始化列表?

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

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

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

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

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

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

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

对比示例:

// 使用初始化列表(高效)
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;
python    int y;
    int distance;

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

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

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

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

总结

初始化列表总结:

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

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

以上为个人经验,希望能给大家一个参考,也希望大家多多支持China编程(www.chinasem.cn)。

这篇关于C++类和对象之初始化列表的使用方式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1154587

相关文章

IDEA之MyBatisX使用的图文步骤

《IDEA之MyBatisX使用的图文步骤》本文主要介绍了IDEA之MyBatisX使用,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习... 目录一、idea插件安装二、IDEA配置数据库连接(以mysql为例)三、生产基础代码一、idea插

Python的pip在命令行无法使用问题的解决方法

《Python的pip在命令行无法使用问题的解决方法》PIP是通用的Python包管理工具,提供了对Python包的查找、下载、安装、卸载、更新等功能,安装诸如Pygame、Pymysql等Pyt... 目录前言一. pip是什么?二. 为什么无法使用?1. 当我们在命令行输入指令并回车时,一般主要是出现以

C++迭代器失效的避坑指南

《C++迭代器失效的避坑指南》在C++中,迭代器(iterator)是一种类似指针的对象,用于遍历STL容器(如vector、list、map等),迭代器失效是指在对容器进行某些操作后... 目录1. 什么是迭代器失效?2. 哪些操作会导致迭代器失效?2.1 vector 的插入操作(push_back,

Java使用WebView实现桌面程序的技术指南

《Java使用WebView实现桌面程序的技术指南》在现代软件开发中,许多应用需要在桌面程序中嵌入Web页面,例如,你可能需要在Java桌面应用中嵌入一部分Web前端,或者加载一个HTML5界面以增强... 目录1、简述2、WebView 特点3、搭建 WebView 示例3.1 添加 JavaFX 依赖3

防止SpringBoot程序崩溃的几种方式汇总

《防止SpringBoot程序崩溃的几种方式汇总》本文总结了8种防止SpringBoot程序崩溃的方法,包括全局异常处理、try-catch、断路器、资源限制、监控、优雅停机、健康检查和数据库连接池配... 目录1. 全局异常处理2. 使用 try-catch 捕获异常3. 使用断路器4. 设置最大内存和线

Java Jackson核心注解使用详解

《JavaJackson核心注解使用详解》:本文主要介绍JavaJackson核心注解的使用,​​Jackson核心注解​​用于控制Java对象与JSON之间的序列化、反序列化行为,简化字段映射... 目录前言一、@jsonProperty-指定JSON字段名二、@JsonIgnore-忽略字段三、@Jso

MySQL中隔离级别的使用详解

《MySQL中隔离级别的使用详解》:本文主要介绍MySQL中隔离级别的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录引言undo log的作用MVCC的实现有以下几个重要因素如何根据这些因素判断数据值?可重复读和已提交读区别?串行化隔离级别的实现幻读和可

使用Python和SQLAlchemy实现高效的邮件发送系统

《使用Python和SQLAlchemy实现高效的邮件发送系统》在现代Web应用中,邮件通知是不可或缺的功能之一,无论是订单确认、文件处理结果通知,还是系统告警,邮件都是最常用的通信方式之一,本文将详... 目录引言1. 需求分析2. 数据库设计2.1 User 表(存储用户信息)2.2 CustomerO

9个SpringBoot中的自带实用过滤器使用详解

《9个SpringBoot中的自带实用过滤器使用详解》在SpringBoot应用中,过滤器(Filter)是处理HTTP请求和响应的重要组件,SpringBoot自带了许多实用的过滤器,如字符编码,跨... 目录1. CharacterEncodingFilter - 字符编码过滤器功能和配置手动配置示例2

Redis持久化机制之RDB与AOF的使用

《Redis持久化机制之RDB与AOF的使用》:本文主要介绍Redis持久化机制之RDB与AOF的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Redis持久化机制-RDB与AOF一、RDB持久化机制1、RDB简介2、RDB的工作原理3、RDB的优缺点4