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

C++类对象所占内存空间大小分析

前言

        类占内存空间是只类实例化后占用内存空间的大小,类本身是不会占内存空间的。用 sizeof 计算类的大小时,实际上是计算该类实例化后对象的大小。空类占用1字节原因:C++要求每个实例在内存中都有一个唯一地址,为了达到这个目的,编译器会给空类隐含添加1字节,保证空类实例化后在内存中得到的地址是独一无二的。

        在C++里空类占的存储空间是0吗?类的成员函数占存储空间吗?类的虚成员函数占存储空间吗?如果对这几个问题的回答不是很确定的话,此篇内容可供参考。

1.空类占用1个字节的存储空间

#include <iostream>
using namespace std;class A {};int main() {cout << "类A所占空间大小:" << sizeof(A) << "byte" << endl;// 定义两个对象a1,a2A a1;A a2;cout << "由类A实例化后的对象a1的地址:" << &a1 << endl;cout << "由类A实例化后的对象a2的地址:" << &a2 << endl;system("pause");return 0;
}

输出结果: ( 每次运行程序时系统为对象 a1、a2 分配的地址不唯一)

原因:类中没有任何成员变量,占用的存储大小本该为0,但是如果是0,类实例化出的对象就不会在内存上占用空间,没有地址,也就无法区分这些对象。为了解决这个问题,编译器会给空类隐含加一个字节,保证用此类定义的对象都有一个独一无二的地址。 

2.类中的普通变量占用存储空间

#include <iostream>
using namespace std;class A {int a; // int类型变量achar p; // char类型变量p
};int main() {cout << "类A所占空间大小:" << sizeof(A) << "byte" << endl;system("pause");return 0;
}

输出结果: 

 

        记得对齐的问题,这点和 struct 的对齐原则很像!int 占 4 字节,char 占 1 字节,补齐 3 字节。因此类 A 占8字节!

3.类的成员函数(非虚函数)不占用存储空间

在1的基础上增加类的成员函数做测试:

#include <iostream>
using namespace std;class A {
public:A(){} // 构造函数~A(){} // 析构函数int func1(){ return 0;} // 普通成员函数int func2(){ return 0;} // 普通成员函数
};int main(){cout << "类A所占空间大小:" << sizeof(A) << "byte" << endl;return 0;
}

输出结果: 

原因:成员函数(包括构造和析构函数)编译后存放在代码区,不占用类的存储空间。

4.类的静态成员变量不占用存储空间

在3的基础上定义一个静态成员变量做测试:

#include <iostream>
using namespace std;class A {
public:A() {} // 构造函数~A(){} // 析构函数int func1() { return 0; } // 普通成员函数int func2() { return 0; } // 普通成员函数private:static int num; // 类的静态成员变量
};int main() {cout << "类A所占空间大小:" << sizeof(A) << "byte" << endl;system("pause");return 0;
}

输出结果: 

原因:类里的静态成员变量在全局数据区中分配空间,不占用类的存储空间,全局只有一份,不会随着类的实例化存储在每个对象里。 

5.类中的虚函数占用存储空间,但所占的空间不会随着虚函数的个数增长

在4的基础上定义3个虚成员函数做测试:

#include <iostream>
using namespace std;class A {
public:A() {} // 构造函数~A(){} // 析构函数int func1() { return 0; } // 普通成员函数int func2() { return 0; } // 普通成员函数// 3个虚函数virtual int func3() { return 0; }virtual int func4() { return 0; }virtual int func5() { return 0; }virtual int func6() { return 0; }
private:static int num; // 类的静态成员变量
};int main() {cout << "类A所占空间大小:" << sizeof(A) << "byte" << endl;system("pause");return 0;
}

输出结果:

(1) x86<32位> 情况下:

(2)x64<64位> 情况下:

原因:C++ 类中有虚函数的时候有一个指向虚函数的指针,在 32 位系统分配指针大小为 4 字节,而在 64 位系统分配指针大小为 8 字节。无论多少个虚函数,只有这一个指针,4 字节(32位系统)或者8字节(64位系统)。注意一般的函数是没有这个指针的,而且也不占类的内存。

6.继承 — 子类所占空间

#include<iostream>
class CBase // 基类
{
public:CBase(void); // 构造函数不占空间virtual ~CBase(void); // 虚析构函数占空间,所占空间根据系统位数而定private:int  a; // 普通变量占空间(4字节)char* p; // 指针类型变量占空间,根据系统位数而定
};class CChild : public CBase // 子类CChild继承Cbase
{
public:CChild(void); // 不占空间~CChild(void); // 不占空间virtual void test();// 父类子类共享一个虚函数指针private:int b;
};int main() {char* str;std::cout << sizeof(str) << std::endl;std::cout << sizeof(CBase) << std::endl;std::cout << sizeof(CChild) << std::endl;
}/*
32位系统下:
4 // 指针类型变量占4字节
12 // 虚析构函数指针占4字节+int a变量占4个字节+char* p占4个字节
16 // int类型变量b占4个字节+基类所占的12个字节// virtual void test();此时不占空间,原因是:父类子类共享一个虚函数指针
*//*
64位系统下:
8 // 指针类型变量占8字节
24 // 虚析构函数指针占8字节+(int a变量占4个字节+对齐另需4字节)+(char* p占4个字节+对齐另需4字节)
32 // (int类型变量b占4个字节+对齐另需4个字节)+基类所占的24个字节
*/

        可见子类的大小是本身成员变量的大小加上父类的大小。其中有一部分是虚函数表的原因,父类子类共享一个虚函数指针。

7.空类与多重继承的空类占用内存空间

#include <iostream>using namespace std;class A {};class A2 {};class B : public A {};class C : public A, public A2 {};class D : public virtual B {}; // 虚继承int main()
{cout << sizeof(A) << endl;cout << sizeof(B) << endl;cout << sizeof(C) << endl;cout << sizeof(D) << endl;return 0;}// 32位系统输出:
/*
1
1
1
4
*/// 64位系统输出:
/*
1
1
1
8
*/

        空类所占内存空间为1;单一继承或多重继承空类的空类所占空间还是1;但虚继承涉及虚指针,指针大小为4(32位系统),故虚继承后空类所占空间为4(32位系统)。

 8.单一继承或多重继承时类占用内存空间

#include <iostream>using namespace std;class A {};class A1 {};class B : public A {int b;
};class C : public A, public A1 {int c;
};int main()
{cout << sizeof(A) << endl;cout << sizeof(B) << endl;cout << sizeof(C) << endl;return 0;}// 32位系统输出:
/*
1
4
8
*/// 64位系统输出:
/*
1
4
8
*/

9.共有继承

#include <iostream>class A {
};class A1 : public A {
};class B : public A{virtual void fun() = 0; // 定义虚函数
};// 共有继承,共用虚函数指针,没有虚基指针
class C : public B{
};class D : public A, public B{
};int main()
{std::cout << "sizeof(A):" << sizeof(A) << std::endl;std::cout << "sizeof(A1):" << sizeof(A1) << std::endl;std::cout << "sizeof(B):" << sizeof(B) << std::endl; std::cout << "sizeof(C):" << sizeof(C) << std::endl; std::cout << "sizeof(D):" << sizeof(D) << std::endl; return 0;
}/*
32位系统下输出:
sizeof(A):1 // 空类A(1)
sizeof(A1):1 // 空类A(0) + A1(1)
sizeof(B):4 // 空类A(0) + 虚函数指针(4)
sizeof(C):4 // 与B共用虚函数指针(4)
sizeof(D):8 // A(1+3<对齐>) + 与B共用虚函数指针(4)
*//*
64位系统下输出:
sizeof(A):1 // 空类A(1)
sizeof(A1):1 // 空类A(0) + A1(1)
sizeof(B):8 // 空类A(0) + 虚函数指针(8)
sizeof(C):8 // 与B共用虚函数指针(8)
sizeof(D):16 // A(1+7<对齐>) + 与B共用虚函数指针(8)
*/

共有继承,共用虚函数指针,没有虚基指针。 

10.虚继承

#include<iostream>/*
虚继承与继承的区别:
1.多了一个虚基指针
2.虚基类位于派生类存储空间的最末尾
3.不会共用虚函数指针
*/class A
{char a[3];
public:virtual void fun1() {};
};// 测试一:单个虚继承,不带虚函数
class B : public virtual A
{char b[3];
};// 测试二:单个虚继承,带自己的虚函数
class C : public virtual A
{char c[3];
public:virtual void fun2() {};
};// 测试三:双重继承
class D : public virtual C
{char d[3];
public:virtual void fun3() {};
};int main()
{std::cout << sizeof(A) << std::endl; std::cout << sizeof(B) << std::endl; std::cout << sizeof(C) << std::endl; std::cout << sizeof(D) << std::endl; return 0;
}/*
32位系统输出:
8 // 8【虚函数指针占4个字节;char a[3]占3个字节,跟虚函数指针所占空间对齐需要另外加1个】
16 // 8(A) + 8(B)【8 == (3+1)+虚基指针】
20 // 8(A) + 12(C)【12 == (3+1)+自己的虚函数指针+虚基指针】
32 // (char d[3]占3个字节+对齐另需1个字节)+类D自己的虚函数指针(4个字节)+虚基指针(4个字节)+(char c[3]占3个字节+对齐另需1个字节)+类C自己的虚函数指针(4个字节)+虚基指针(4个字节)+(char a[3]占3个字节,跟虚函数指针所占空间对齐需要另外加1个字节) + 类A自己的虚函数指针占4个字节*//*
64位系统输出:
16 // 【虚函数指针占8个字节;char a[3]占3个字节,跟虚函数指针所占空间对齐需要另外加5个】
32 // 16(A) + 16【16 == (char b[3]占3个字节+对齐另需5个字节)+虚基指针(8个字节)】
40 // 16(A) + 24【24 == <(char c[3]占3个字节+对齐另需5个字节)+自己的虚函数指针(8个字节)+虚基指针(8个字节)>】
64 // (char d[3]占3个字节+对齐另需5个字节)+类D自己的虚函数指针(8个字节)+虚基指针(8个字节)+(char c[3]占3个字节+对齐另需5个字节)+类C自己的虚函数指针(8个字节)+虚基指针(8个字节)+(char a[3]占3个字节,跟虚函数指针所占空间对齐需要另外加5个字节) + 类A自己的虚函数指针占8个字节
*/

        注意,虚继承的时候 A B C D 四个类不仅不会共享虚基类指针,也不会共享虚函数指针,要和普通继承区分开来。

具体分析如下:

 class A size(8):+---0    | {vfptr}4    | a| <alignment member> (size=1)8    +---class B size(16):+---0    | {vfptr}4    | {vbptr}8    | b| <alignment member> (size=1)+---+--- (virtual base A)
12    | a| <alignment member> (size=1)
16    +---class C size(20):+---0    | {vfptr}4    | {vbptr}8    | b| <alignment member> (size=1)+---+--- (virtual base A)
12    | {vfptr}
16    | a| <alignment member> (size=1)
20    +---    class D size(32):+---0    | {vfptr}4    | {vbptr}8    | c| <alignment member> (size=1)+---+--- (virtual base A)
12    | {vfptr}
16    | a| <alignment member> (size=1)+---+--- (virtual base B)
20    | {vfptr}
24    | {vbptr}
28    | b| <alignment member> (size=1)
32    +---
  • 虚表(vftable
  • 虚函数指针(vfptr
  • 虚基指针(vbptr

11.总结

        空的类是会占用内存空间的,而且大小是 1,原因是 C++ 要求每个实例(对象)在内存中都有独一无二的地址。

(一)类内部的成员变量:

  • 普通的变量:是要占用内存的,但是要注意对齐原则(这点和 struct 类型很相似)。
  • static 修饰的静态变量:不占用内容,原因是编译器将其放在全局变量区。

(二)类内部的成员函数:

  • 普通函数:不占用内存。
  • 虚函数:有一个指向虚函数的指针,要占用 4 个字节或 8 个字节(根据系统位数来定),用来指定虚函数的虚拟函数表的入口地址。所以一个类的虚函数所占用的地址是不变的,和虚函数的个数是没有关系的。

(三)虚继承与继承的区别:

  • 多了一个虚基指针。
  • 虚基类位于派生类存储空间的最末尾。
  • 不会共用虚函数指针。

参考自:

C++类对象到底占多大存储空间呢_类的成员函数占用空间吗_haowunanhai的博客-CSDN博客
https://www.cnblogs.com/linuxAndMcu/p/10388330.html

C++中的类所占内存空间总结

c++虚表(vftable)、虚函数指针(vfptr)、虚基指针(vbptr)的测试结果

https://www.cnblogs.com/aqing1987/p/4210773.html 

相关文章:

C++类对象所占内存空间大小分析

前言 类占内存空间是只类实例化后占用内存空间的大小&#xff0c;类本身是不会占内存空间的。用 sizeof 计算类的大小时&#xff0c;实际上是计算该类实例化后对象的大小。空类占用1字节原因&#xff1a;C要求每个实例在内存中都有一个唯一地址&#xff0c;为了达到这个目的&am…...

绿肥红瘦专栏数据的爬取

前言 要想爬专栏&#xff0c;先得爬用户。要想爬用户&#xff0c;三个header参数挡住了去路&#xff1a;x-zst-81&#xff0c;x-zse-93&#xff0c;x-zse-96&#xff0c;经过搜索x-zse-96&#xff0c;定位到设置该字段的位置&#xff1a; 这个t2是固定的值&#xff0c;t0来自于…...

centos或aws linux部署java应用,环境搭建shell

目录 设置root密码开启密码登录安装docker安装nginx设置nginx自启动nginx配置https配置http集群tcp集群 安装docker设置docker自启动修改docker基础配置创建docker网关docker安装mysql单机版本主从版本 docker安装redis设置密码&#xff1a;不要密码&#xff1a; docker安装rab…...

2023年中国车用冲压模具行业特征、竞争现状及行业市场规模分析[图]

汽车冲压件模具具有尺寸大、型面复杂、精度要求高等特点&#xff0c;属于技术密集型产品。汽车冲压模具能快速精密地把材料直接加工成零件或半成品并通过焊接、铆接、拼装等工艺装配成零部件&#xff0c;冲压模具的设计开发和加工能力对汽车冲压零部件产品总制造成本、质量及性…...

基于Pytorch的CNN手写数字识别

作为深度学习小白&#xff0c;我想把自己学习的过程记录下来&#xff0c;作为实践部分&#xff0c;我会写一个通用框架&#xff0c;并会不断完善这个框架&#xff0c;作为自己的入门学习。因此略过环境搭建和基础知识的步骤&#xff0c;直接从代码实战开始。 一.下载数据集并加…...

Java设计模式之观察者模式(Observer Pattern)

观察者模式&#xff08;Observer Pattern&#xff09;是一种常用的软件设计模式&#xff0c;它用于在对象之间建立一种一对多的依赖关系&#xff0c;当一个对象的状态发生变化时&#xff0c;它的所有依赖对象都会得到通知并自动更新。观察者模式属于行为型模式。 在观察者模式…...

最优化:建模、算法与理论(最优性理论2

5.7 约束优化最优性理论应用实例 5.7.1 仿射空间的投影问题 考虑优化问题 min ⁡ x ∈ R n 1 2 ∣ ∣ x − y ∣ ∣ 2 2 , s . t . A x b \min_{x{\in}R^n}\frac{1}{2}||x-y||_2^2,\\ s.t.{\quad}Axb x∈Rnmin​21​∣∣x−y∣∣22​,s.t.Axb 其中 A ∈ R m n , b ∈ R m …...

redis一主一从搭建

1.复制一份redis.conf并将6380都改成6379 [redist3-dtpoc-dtpoc-web06 conf]$ cp redis.conf redis_6380.conf [redist3-dtpoc-dtpoc-web06 conf]$ vi redis_6380.conf port 6380 daemonize yes pidfile "/home/redis/redis/logs/redis_6380.pid" logfile "/hom…...

【MySql】8- 实践篇(六)

文章目录 1. MySql保证主备一致1.1 MySQL 主备的基本原理1.2 binlog 的三种格式对比1.3 循环复制问题 2. MySql保证高可用2.1 主备延迟2.2 主备延迟的来源2.3 可靠性优先策略2.4 可用性优先策略 3. 备库为何会延迟很久-备库并行复制能力3.1 MySQL 5.6 版本的并行复制策略3.2 Ma…...

Spring篇---第七篇

系列文章目录 文章目录 系列文章目录一、说说事务的传播级别二、Spring 事务实现方式三、Spring框架的事务管理有哪些优点一、说说事务的传播级别 Spring事务定义了7种传播机制: PROPAGATION_REQUIRED:默认的Spring事物传播级别,若当前存在事务,则加入该事务,若 不存在事务…...

2023年中国轮胎模具需求量、竞争格局及行业市场规模分析[图]

轮胎模具是轮胎生产线中的硫化成形装备&#xff0c;是高技术含量、高精度及高附加值的个性化模具产品&#xff0c;尤其是轮胎的花纹、图案、字体以及其他外观特征的成形都依赖于轮胎模具&#xff0c;因此其制造技术难度较高。其主要功能是通过所成型材料&#xff08;主要是橡塑…...

集成学习方法(随机森林和AdaBoost)

释义 集成学习很好的避免了单一学习模型带来的过拟合问题 根据个体学习器的生成方式&#xff0c;目前的集成学习方法大致可分为两大类&#xff1a; Bagging(个体学习器间不存在强依赖关系、可同时生成的并行化方法) 流行版本&#xff1a;随机森林(random forest)Boosting(个体…...

PeopleCode中Date函数的用法

语法 Date(date_num) 描述 The Date function takes a number in the form YYYYMMDD and returns a corresponding Date value. If the date is invalid, Date displays an error message. Date函数输入是一个形如“YYYYMMDD”的数字&#xff0c;返回一个相应的Date类型的值…...

解决 el-tree setChecked 方法偶尔失效的方法

目前在大多数公司中&#xff0c;菜单的权限控制都是不可或缺的功能 在和后端配合做权限控制的时候不可避免的会用到 el-tree 然而这个组件本身带的坑不少 我们需要回显对应角色拥有的菜单&#xff0c;在不严格的模式下&#xff0c;父节点的选中会连带子节点的选中 如果 &a…...

重磅发布!RflySim Cloud 智能算法云仿真平台亮相,助力大规模集群算法高效训练

RflySim Cloud智能算法云仿真平台&#xff08;以下简称RflySim Cloud平台&#xff09;是由卓翼智能及飞思实验室为无人平台集群算法验证、大规模博弈对抗仿真、人工智能模型训练等前沿研究领域研发的平台。主要由环境仿真模块、物理效应计算模块、多智能体仿真模块、分布式网络…...

C++ 01.学习C++的意义-狄泰软件学院

一些历史 UNIX操作系统诞生之初是用汇编语言编写的随着UNIX系统的发展&#xff0c;汇编语言的开发效率成为瓶颈&#xff0c;所以需要一个新的语言替代汇编语言1971年通过对B语言改良&#xff0c;使其能直接产生机器代码&#xff0c;C语言诞生UNIX使用C语言重写&#xff0c;同时…...

微软正式发布开源应用平台 Radius平台

“ 10 月 18 日&#xff0c;微软 Azure 孵化团队正式发布开源应用平台 Radius&#xff0c;该平台将应用程序置于每个开发阶段的中心&#xff0c;重新定义应用程序的构建、管理与理解方式。” 简单的概括就是&#xff0c;它和Kubernetes不一样&#xff0c;Radius将应用程序放在每…...

排序算法(python)

排序算法 冒泡排序 一次比较相邻的两个数&#xff0c;每轮之后末尾的数字是确定的。 时间复杂度为 O ( n 2 ) O(n^2) O(n2)&#xff0c;空间复杂度为 O ( 1 ) O(1) O(1)&#xff0c;稳定。 def BUB(nums):for i in range(len(nums)):count 0for j in range(len(nums)-i-1)…...

一款简单漂亮的WPF UI - AduSkin

前言 经常会有同学会问&#xff0c;有没有好看简单的WPF UI库推荐的。今天就给大家推荐一款简单漂亮的WPF UI&#xff0c;融合多个开源框架组件&#xff1a;AduSkin。 WPF是什么&#xff1f; WPF 是一个强大的桌面应用程序框架&#xff0c;用于构建具有丰富用户界面的 Windo…...

Java面试题-Java核心基础-第七天(String)

目录 一、String、StringBuffer、StringBuilder的区别 二、String为什么是不可变的 三、字符串拼接用""还是用StringBuilder 四、String 中的equals和Object中的equals的区别 五、字符串常量池的作用了解吗&#xff1f; 六、String s1 new String("abc&qu…...

web vue 项目 Docker化部署

Web 项目 Docker 化部署详细教程 目录 Web 项目 Docker 化部署概述Dockerfile 详解 构建阶段生产阶段 构建和运行 Docker 镜像 1. Web 项目 Docker 化部署概述 Docker 化部署的主要步骤分为以下几个阶段&#xff1a; 构建阶段&#xff08;Build Stage&#xff09;&#xff1a…...

【kafka】Golang实现分布式Masscan任务调度系统

要求&#xff1a; 输出两个程序&#xff0c;一个命令行程序&#xff08;命令行参数用flag&#xff09;和一个服务端程序。 命令行程序支持通过命令行参数配置下发IP或IP段、端口、扫描带宽&#xff0c;然后将消息推送到kafka里面。 服务端程序&#xff1a; 从kafka消费者接收…...

大话软工笔记—需求分析概述

需求分析&#xff0c;就是要对需求调研收集到的资料信息逐个地进行拆分、研究&#xff0c;从大量的不确定“需求”中确定出哪些需求最终要转换为确定的“功能需求”。 需求分析的作用非常重要&#xff0c;后续设计的依据主要来自于需求分析的成果&#xff0c;包括: 项目的目的…...

LeetCode - 394. 字符串解码

题目 394. 字符串解码 - 力扣&#xff08;LeetCode&#xff09; 思路 使用两个栈&#xff1a;一个存储重复次数&#xff0c;一个存储字符串 遍历输入字符串&#xff1a; 数字处理&#xff1a;遇到数字时&#xff0c;累积计算重复次数左括号处理&#xff1a;保存当前状态&a…...

最新SpringBoot+SpringCloud+Nacos微服务框架分享

文章目录 前言一、服务规划二、架构核心1.cloud的pom2.gateway的异常handler3.gateway的filter4、admin的pom5、admin的登录核心 三、code-helper分享总结 前言 最近有个活蛮赶的&#xff0c;根据Excel列的需求预估的工时直接打骨折&#xff0c;不要问我为什么&#xff0c;主要…...

React19源码系列之 事件插件系统

事件类别 事件类型 定义 文档 Event Event 接口表示在 EventTarget 上出现的事件。 Event - Web API | MDN UIEvent UIEvent 接口表示简单的用户界面事件。 UIEvent - Web API | MDN KeyboardEvent KeyboardEvent 对象描述了用户与键盘的交互。 KeyboardEvent - Web…...

OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别

OpenPrompt 和直接对提示词的嵌入向量进行训练有什么区别 直接训练提示词嵌入向量的核心区别 您提到的代码: prompt_embedding = initial_embedding.clone().requires_grad_(True) optimizer = torch.optim.Adam([prompt_embedding...

MySQL 知识小结(一)

一、my.cnf配置详解 我们知道安装MySQL有两种方式来安装咱们的MySQL数据库&#xff0c;分别是二进制安装编译数据库或者使用三方yum来进行安装,第三方yum的安装相对于二进制压缩包的安装更快捷&#xff0c;但是文件存放起来数据比较冗余&#xff0c;用二进制能够更好管理咱们M…...

day36-多路IO复用

一、基本概念 &#xff08;服务器多客户端模型&#xff09; 定义&#xff1a;单线程或单进程同时监测若干个文件描述符是否可以执行IO操作的能力 作用&#xff1a;应用程序通常需要处理来自多条事件流中的事件&#xff0c;比如我现在用的电脑&#xff0c;需要同时处理键盘鼠标…...

《Docker》架构

文章目录 架构模式单机架构应用数据分离架构应用服务器集群架构读写分离/主从分离架构冷热分离架构垂直分库架构微服务架构容器编排架构什么是容器&#xff0c;docker&#xff0c;镜像&#xff0c;k8s 架构模式 单机架构 单机架构其实就是应用服务器和单机服务器都部署在同一…...