当前位置: 首页 > news >正文

编译器自动生成的构造函数

背景

作为一个C++小白,最近在看深度解析对象模型的时候,发现一个很久以来的认知错误:编译器会为没有定义构造函数的class生成一个默认构造函数。其实这个观点是错误的,编译器只会在四种情况下生成。

相关知识
  • 一定要明确一个事情,就是基础数据类型没有默认构造函数,因为int, float,char这些不是class/struct,基础数据类型就是基础数据类型,不要自己强行脑补为类或者结构。
  • 构造函数花括号里不能用小括号初始化任何成员变量,凡是想初始化都放在初始化列表里就好了
四种情况
情况一

当成员变量中有默认构造函数时,举例说明:

class Test
{
public:int a;float b;
};
int main()
{Test t;int c = t.a +1;return 0;
}

在这里插入图片描述

这种情况下编译器不会生成默认构造函数,因为int, float不是对象,没有默认构造函数。可以根据汇编代码发现确实没有。

class Test
{
public:int a;float b;string s;
};

在这里插入图片描述
可以看到一旦加入string的话就会出现默认构造函数了。

情况二

该类的基类有默认构造函数

情况三

类中存在虚函数,那么C++编译器会为你生成默认构造函数,以初始化虚表(虚函数表vftable)。

情况四

存在虚基类,那么C++编译器会为你生成默认构造函数,以初始化虚基类表(vbtable)。

生成默认构造函数
  1. 首先生成一个空壳子
inline Construct()
{}
  1. 补充初始化列表

这里是一个知识点,list中的顺序是按声明顺序来的,跟你怎么写代码没啥关系

这里初始化列表会把所有的成员变量的默认构造函数依次加入进来,下面是一个伪代码的例子:

class Test
{
public:float d;string A;Student B;Teacher C;int e;
}
inline Test() {A(), B(), C(), e} //这里e不会初始化,只是分配内存
{}
  1. 把初始化列表挪到函数大括号中
inline Test()
{A();B();C();e;
}

**可以看到,自动生成的默认构造函数其实并不会给所有成员都初始化**。此外,这里可能会觉得第三步和第二步可以合在一起,其实拆开是为了后面程序员自己添加构造函数时,编译器补充函数信息逻辑更清楚。

自定义默认构造函数

对于自定义默认构造函数,编译器有三个原则:

  • 插入的代码只能在用户实现的代码前
  • 补充的代码符合成员变量的声明顺序
  • 基础数据类型依然不是编译器的考虑的范围
方式一、初始化列表为空

我们根据上面已经知道,编译器会自动生成构造函数,但是如果程序员自己提供了构造函数怎么处理?

class Test
{
public:float d;string A;Student B;Teacher C;int e;
}
Test()
Test::Test()
{B("student");
}
  1. 补充初始化列表
Test()
Test::Test(){d, A(), B(), C(), e}
{//B("student"); //编译器会报错:成员变量**对象**(不是基础数据)必须在构造函数初始化列表里初始化B = "student";
}
  1. 挪到函数中
Test()
Test::Test(){}
{d;A();B();C();e;B = "student";// 等效于:tmp("student"), B=tmp;
}
方式一、初始化列表不为空
class Test
{
public:float d;string A;Student B;Teacher C;int e;
}
Test()
Test::Test():B("student")
{
}
  1. 补充初始化列表
Test()
Test::Test(){d, A(), B("student"), C(), e}
{
}
  1. 挪到函数中
Test()
Test::Test(){}
{d;A();B("student");C();e;
}

根据这两个小例子可以好好的体会上面的三个原则

http://www.lryc.cn/news/174967.html

相关文章:

  • SpringSecurity - 认证与授权、自定义失败处理、跨域问题、认证成功/失败处理器
  • 自定义映射resultMap
  • Android修行手册 - Android Studio去掉方法参数提示、变量类型提示、方法引用Usage提示
  • 【车载开发系列】ECU Application Software程序刷新步骤
  • inject和provide的使用
  • 2023年中国研究生数学建模竞赛D题
  • Unity制作曲线进度条
  • 面试:C++ 11 智能指针
  • 设计模式——3. 抽象工厂模式
  • vscode 无法使用 compilerPath“D:.../bin/arm-none-eabi-g++.exe”解析配置。
  • Vue.js入门模板语法[上] 及Vue.js实现购物车---详细讲解
  • windows下gvim的配置
  • 基于复旦微的FMQL45T900全国产化ARM开发开发套件(核心板+底板)
  • Leetcode Top100(23)环形链表
  • 线性代数基础-行列式
  • RT-Thread(学习)
  • 【MySQL】 MySQL 死锁问题分析优化器特性及优化方案
  • 【C++面向对象侯捷】8.栈,堆和内存管理
  • 在比特币上使用可检索性证明支付存储费用
  • 使用SSE(Server-Sent Events)实现服务端给客户端发消息
  • 【Redis】使用rpm包安装redis
  • 论文阅读-Group-based Fraud Detection Network on e-Commerce Platforms
  • java程序启动时指定JVM内存参数和Xms、Xmx参数学习
  • 【C++编程能力提升】
  • FlashDuty Changelog 2023-09-21 | 自定义字段和开发者中心
  • 贪心算法-
  • 漫谈:C语言 C++ 左值、右值、类型转换
  • 前车之鉴,后车之师
  • WEB使用VUE3实现地图导航跳转
  • 今天聊一聊高性能系统架构设计是什么样的