C++类和对象之初始化列表

初始化列表

  • C++初始化列表详解:性能优化与正确实践
    • 什么是初始化列表?
    • 初始化列表的三大核心作用
      • 1. 性能优化:避免不必要的赋值操作
      • 2. 强制初始化:处理const和引用成员
      • 3. 基类初始化:正确调用父类构造函数
      • 4.必须使用初始化列表的情况
        • 1. 引用(Reference)
        • 2. `const` 成员变量
        • 3. 没有默认构造函数的类类型成员
        • 为什么不能在构造函数体中初始化?
    • 初始化列表的语法规则
      • 1. 初始化顺序由声明顺序决定
      • 2. 支持表达式初始化
    • 初始化列表 vs 构造函数体赋值
    • 总结

在这里插入图片描述

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/引用成员支持不支持
基类初始化必须使用不可用

总结

初始化列表总结:

  1. ⽆论是否显⽰写初始化列表,每个构造函数都有初始化列表;
  2. ⽆论是否在初始化列表显⽰初始化成员变量,每个成员变量都要⾛初始化列表初始化;在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/bicheng/80676.shtml
繁体地址,请注明出处:http://hk.pswp.cn/bicheng/80676.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

continue通过我们的开源 IDE 扩展和模型、规则、提示、文档和其他构建块中心,创建、共享和使用自定义 AI 代码助手

​一、软件介绍 文末提供程序和源码下载 Continue 使开发人员能够通过我们的开源 VS Code 和 JetBrains 扩展以及模型、规则、提示、文档和其他构建块的中心创建、共享和使用自定义 AI 代码助手。 二、功能 Chat 聊天 Chat makes it easy to ask for help from an LLM without…

基于Spring Boot + Vue的母婴商城系统( 前后端分离)

一、项目背景介绍 随着母婴行业在互联网平台的快速发展,越来越多的家庭倾向于在线选购母婴产品。为了提高商品管理效率和用户购物体验,本项目开发了一个基于 Spring Boot Vue 技术栈的母婴商城系统,实现了商品分类、商品浏览、资讯展示、评…

实战演练:用 AWS Lambda 和 API Gateway 构建你的第一个 Serverless API

实战演练:用 AWS Lambda 和 API Gateway 构建你的第一个 Serverless API 理论千遍,不如动手一遍!在前面几篇文章中,我们了解了 Serverless 的概念、FaaS 的核心原理以及 BaaS 的重要作用。现在,是时候把这些知识运用起来,亲手构建一个简单但完整的 Serverless 应用了。 …

node.js 实战——express图片保存到本地或服务器(七牛云、腾讯云、阿里云)

本地 ✅ 使用formidable 读取表单内容 npm i formidable ✅ 使用mime-types 获取图片后缀 npm install mime-types✅ js 中提交form表单 document.getElementById(uploadForm).addEventListener(submit, function(e){e.preventDefault();const blob preview._blob;if(!blob)…

2025最新:3分钟使用Docker快速部署单节点Redis

🧑‍🏫 详细教程:通过 Docker 安装单节点 Redis 🛠️ 前提条件: 你需要在 Ubuntu 系统上进行操作(如果你在其他系统上操作,可以按相似步骤进行调整)。已安装 Docker 和 Docker Com…

CentOS 7 系统下安装 OpenSSL 1.0.2k 依赖问题的处理

前面有提到过这个openssl的版本冲突问题,也是在这次恢复服务器时遇到的问题,我整理如下,供大家参考。小小一个软件的安装,挺坑的。 一、问题 项目运行环境需要,指定PHP7.0.9这个版本,但是‌系统版本与软件…

LoRA(Low-Rank Adaptation)原理详解

LoRA(Low-Rank Adaptation)原理详解 LoRA(低秩适应)是一种参数高效微调(Parameter-Efficient Fine-Tuning, PEFT)技术,旨在以极低的参数量实现大模型在特定任务上的高效适配。其核心思想基于低秩分解假设,即模型在适应新任务时,参数更新矩阵具有低秩特性,可用少量参…

Solana批量转账教程:提高代币持有地址和生态用户空投代币

前言 Solana区块链因其高吞吐量和低交易费用成为批量操作(如空投)的理想选择。本教程将介绍几种在Solana上进行批量转账的方法,帮助您高效地向多个地址空投代币。 solana 账户模型 在Solana中有三类账户: 数据账户,…

基于LSTM与SHAP可解释性分析的神经网络回归预测模型【MATLAB】

基于LSTM与SHAP可解释性分析的神经网络回归预测模型【MATLAB】 一、引言 在数据驱动的智能时代,时间序列预测已成为许多领域(如金融、气象、工业监测等)中的关键任务。长短期记忆网络(LSTM)因其在捕捉时间序列长期依…

手机网页提示ip被拉黑名单什么意思?怎么办

‌当您使用手机浏览网页时,突然看到“您的IP地址已被列入黑名单”的提示,是否感到困惑和不安?这种情况在现代网络生活中并不罕见,但确实会给用户带来诸多不便。本文将详细解释IP被拉黑的含义、常见原因,并提供一系列实…

Java消息队列性能优化实践:从理论到实战

Java消息队列性能优化实践:从理论到实战 1. 引言 在现代分布式系统架构中,消息队列(Message Queue,MQ)已经成为不可或缺的中间件组件。它不仅能够实现系统间的解耦,还能提供异步通信、流量削峰等重要功能…

BUUCTF——Cookie is so stable

BUUCTF——Cookie is so stable 进入靶场 页面有点熟悉 跟之前做过的靶场有点像 先简单看一看靶场信息 有几个功能点 flag.php 随便输了个admin 根据题目提示 应该与cookie有关 抓包看看 构造payload Cookie: PHPSESSIDef0623af2c1a6d2012d57f3529427d52; user{{7*7}}有…

json格式不合法情况下,如何尽量保证数据可用性

背景 在工作流程中,并非所有数据都如人所愿,即使json版本也会由于csv、tsv、excel、text等不同文件格式转化、获取数据源不完整等问题,造成我们要处理的json文件存在不合法。 尝试方案 除了人为修正外,有效的方法是使用json“修…

Python基础总结(十)之函数

Python函数 函数是Python中也是非常重要的,函数是带名字的代码块,用于完成具体的工作。要执行函数定义的特定任务,可调用该函数。 一、函数的定义 函数的定义要使用def关键字,def后面紧跟函数名,缩进的为函数的代码块。 def test():print("Hello,World")上述…

懒人美食帮SpringBoot订餐系统开发实现

概述 快速构建一个订餐系统,今天,我们将通过”懒人美食帮”这个基于SpringBoot的订餐系统项目,为大家详细解析从用户登录到多角色权限管理的完整实现方案。本教程特别适合想要学习企业级应用开发的初学者。 主要内容 1. 用户系统设计与实现…

AI(学习笔记第三课) 使用langchain进行AI开发(2)

文章目录 AI(学习笔记第三课) 使用langchain进行AI开发(2)学习内容:1. 返回结构化数据(structured_output pydantic)1.1 使用背景1.2 返回结构化数据示例代码(pydantic)1.3 执行测试代码2 返回结构化数据(json)2.1 示例代码2.2 执行结果3 给提供一些例子(few shot pr…

unity 使用蓝牙通讯(PC版,非安卓)

BlueTooth in pc with unity 最近接到的需求是在unity里面开发蓝牙功能,其实一开始我并不慌,因为据我所知,unity有丰富的插件可以使用,但是问题随之而来 1.unity里面无法直接与蓝牙通讯(后来找到了开启runtime一类的东西,但是我找了半天也没找到在哪里可以打开) 2.引入dll通过d…

MySQL中的意向锁 + next-key锁 + 间隙锁

引言 在数据库并发控制中,锁机制是保障数据一致性和隔离性的核心手段。MySQL中意向锁、间隙锁以及next-key锁等复杂锁类型,旨在协调表级锁与行级锁之间的关系,防止数据的脏读、不可重复读和幻读现象,尤其是在可重复读隔离级别下发…

机器学习 数据集

数据集 1. scikit-learn工具介绍1.1 scikit-learn安装1.2 Scikit-learn包含的内容 2 数据集2.1 sklearn玩具数据集介绍2.2 sklearn现实世界数据集介绍2.3 sklearn加载玩具数据集示例1:鸢尾花数据示例2:分析糖尿病数据集 2.4 sklearn获取现实世界数据集示…

Linux-c语言串口程序

c语言串口程序 // C library headers #include <stdio.h> #include <string.h>// Linux headers #include <fcntl.h> // Contains file controls like O_RDWR #include <errno.h> // Error integer and strerror() function #include <termios.h&g…