C++中类大小的问题

2024-06-08 05:08
文章标签 c++ 问题 大小 中类

本文主要是介绍C++中类大小的问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

    • 1.C++类大小问题
    • 2.虚继承和虚函数混合使用类大小

1.C++类大小问题

eg:

#include<iostream>
using namespace std;class a {};
class b {};
class c :public a{virtual void fun() = 0;
};
class d :public b, public c{};int main()
{cout << "sizeof(a)" << sizeof(a) << endl;cout << "sizeof(b)" << sizeof(b) << endl;cout << "sizeof(c)" << sizeof(c) << endl;cout << "sizeof(d)" << sizeof(d) << endl;return 0;
}
  • 结果:
    在这里插入图片描述

分析:

  • 类的实例化?
    所谓类的实例化就是在内存中分配一块地址
  • 空类的大小为什么是1?
    这就是我们刚才所说的实例化的原因(空类同样可以被实例化),每个实例在内存中都有一个独一无二的地址,为了达到这个目的,编译器往往会给一个空类隐含的加一个字节,这样空类在实例化后在内存得到了独一无二的地址,所以a,b的大小为1。
  • 类c的大小为什么是4?
    而类c是由类a派生而来,它里面有一个纯虚函数,由于有虚函数的原因,有一个指向虚函数的指针(vfptr),在32位的系统分配给指针的大小为4个字节,所以最后得到c类的大小为4.
  • 类d的大小为什么是8?
    类d是由类b,c派生而来,它的大小应该为二者之和5,为什么却是8呢?
    这是因为为了提高实例在内存中的存取效率.类的大小往往被调整到系统的整数倍。并采取就近的法则,里哪个最近的倍数,就是该类的大小,所以类d的大小为8个字节
  • 结论:
    不管类是否为空类,均可被实例化(空类也可被实例化),每个被实例都有一个独一无二的地址

eg:

#include<iostream>
using namespace std;class a{
private:int data;
};class b{
private:int data;static int data1;
};
int b::data1 = 0;
void main(){cout << "sizeof(a)=" << sizeof(a) << endl;cout << "sizeof(b)=" << sizeof(b) << endl;
}
  • 测试:
    在这里插入图片描述

分析:

  • 为什么类b多了一个数据成员,却大小和类a的大小相同呢?
    管这个类实际产生了多少实例,还是派生了多少新的类,静态成员数据在类中永远只有一个实体存在,而类的非静态数据成员只有被实例化的时候,他们才存在。但是类的静态数据成员一旦被声明,无论类是否被实例化,它都已存在。

eg:

#include<iostream>
using namespace std;class A{
public:A(int a){x = a;}void f(int x){cout << x << endl;}~A(){}private:int x;int g;
};
class B{
public:
private:int data; int data2;static int xs;
};
int B::xs = 0;
void main(){A s(10);s.f(10);cout << "sozeof(a) = " << sizeof(A) << endl;cout << "sizeof(b) = " << sizeof(B) << endl;
}
  • 测试:
    在这里插入图片描述

分析:

  • 它们的结果均相同,可以看出类的大小与它当中的构造函数,析构函数,以及其他的成员函数无关,只与它当中的成员数据有关

结论:

  • 从以上的几个例子不难发现类的大小:
    1.为类的非静态成员数据的类型大小之和.
    2.有编译器额外加入的成员变量的大小,用来支持语言的某些特性(如:指向虚函数的指针).
    3.为了优化存取效率,进行的边缘调整.
    4 与类中的构造函数,析构函数以及其他的成员函数无关.

2.虚继承和虚函数混合使用类大小

eg:

#include<iostream>
using namespace std;class A
{virtual void aa(){};
};class B : public virtual  A
{char j[3];                                    //加入一个变量是为了看清楚class中的vfptr放在什么位置
public:virtual void bb(){};
};class B1 : public A
{char j[3];                                    //加入一个变量是为了看清楚class中的vfptr放在什么位置
public:virtual void bb(){};
};class C : public virtual B
{char i[3];
public:virtual void cc(){};
};void main(){cout << "sozeof(A) = " << sizeof(A) << endl;cout << "sizeof(B) = " << sizeof(B) << endl;cout << "sizeof(B1) = " << sizeof(B1) << endl;cout << "sizeof(C) = " << sizeof(C) << endl;
}
  • 测试:
    在这里插入图片描述

分析:

  • 对于class A
    由于只有一个虚函数,那么必须得有一个对应的虚函数表,来记录对应的函数入口地址。同时在class A的内存空间中之需要有个vfptr_A指向该表。sizeof(A)也很容易确定,为4。

  • 对于class B
    由于class B虚基础了class A,同时还拥有自己的虚函数。
    (1)那么class B中首先拥有一个vfptr_B,指向自己的虚函数表。还有char j[3],做一次alignment,一般大小为4。
    (2)那么虚继承呢?
    首先要通过加入一个虚基类指针(记vbptr_B_A)来指向其父类,然后还要包含父类的所有内容。sizeof(B)= 4+4+4+4=16(vfptr_B、char j[3]做alignment、vbptr_B_A和class A(类A的虚表指针))。

  • 对于类B1
    因为不是虚继承,所以子类不会继承父类的虚表指针,它会有用自己的虚表指针并指向自身的虚表,所以sizeof(B1)=4+4=8(vfptr_B、char j[3]做alignment)。
    而如果是虚继承,则子类会继承父类的虚表指针,一份指向自己的虚表、一份指向虚基表;
    每个类都有自己的虚表和虚表指针、在虚继承时只有子类才有虚机类表指针和虚基类表。
    VC在编译时会把vfptr放到类的头部;

  • 对于Class C
    lass C首先也得有个vfptr_C,然后是char i[3],然后是vbptr_C_B,然后是class B,所以sizeof(C)=4+4+4+16=28(vfptr_C、char i[3]做alignment、vbptr_C_A和class B)

  • 参考:链接

这篇关于C++中类大小的问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


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

相关文章

redis在spring boot中异常退出的问题解决方案

《redis在springboot中异常退出的问题解决方案》:本文主要介绍redis在springboot中异常退出的问题解决方案,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴... 目录问题:解决 问题根源️ 解决方案1. 异步处理 + 提前ACK(关键步骤)2. 调整Redis消费者组

Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题

《Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题》:本文主要介绍Ubuntu上手动安装Go环境并解决“可执行文件格式错误”问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未... 目录一、前言二、系统架构检测三、卸载旧版 Go四、下载并安装正确版本五、配置环境变量六、验证安装七、常见

解决Java异常报错:java.nio.channels.UnresolvedAddressException问题

《解决Java异常报错:java.nio.channels.UnresolvedAddressException问题》:本文主要介绍解决Java异常报错:java.nio.channels.Unr... 目录异常含义可能出现的场景1. 错误的 IP 地址格式2. DNS 解析失败3. 未初始化的地址对象解决

springboot+vue项目怎么解决跨域问题详解

《springboot+vue项目怎么解决跨域问题详解》:本文主要介绍springboot+vue项目怎么解决跨域问题的相关资料,包括前端代理、后端全局配置CORS、注解配置和Nginx反向代理,... 目录1. 前端代理(开发环境推荐)2. 后端全局配置 CORS(生产环境推荐)3. 后端注解配置(按接口

使用雪花算法产生id导致前端精度缺失问题解决方案

《使用雪花算法产生id导致前端精度缺失问题解决方案》雪花算法由Twitter提出,设计目的是生成唯一的、递增的ID,下面:本文主要介绍使用雪花算法产生id导致前端精度缺失问题的解决方案,文中通过代... 目录一、问题根源二、解决方案1. 全局配置Jackson序列化规则2. 实体类必须使用Long封装类3.

Idea插件MybatisX失效的问题解决

《Idea插件MybatisX失效的问题解决》:本文主要介绍Idea插件MybatisX失效的问题解决,详细的介绍了4种问题的解决方法,具有一定的参考价值,感兴趣的可以了解一下... 目录一、重启idea或者卸载重装MyBATis插件(无需多言)二、检查.XML文件与.Java(该文件后缀Idea可能会隐藏

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

《C++类和对象之初始化列表的使用方式》:本文主要介绍C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C++初始化列表详解:性能优化与正确实践什么是初始化列表?初始化列表的三大核心作用1. 性能优化:避免不必要的赋值操作2. 强

Nginx 访问 /root/下 403 Forbidden问题解决

《Nginx访问/root/下403Forbidden问题解决》在使用Nginx作为Web服务器时,可能会遇到403Forbidden错误,文中通过示例代码介绍的非常详细,对大家的学习或者工作... 目录解决 Nginx 访问 /root/test/1.html 403 Forbidden 问题问题复现Ng

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

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

Nginx部署React项目时重定向循环问题的解决方案

《Nginx部署React项目时重定向循环问题的解决方案》Nginx在处理React项目请求时出现重定向循环,通常是由于`try_files`配置错误或`root`路径配置不当导致的,本文给大家详细介... 目录问题原因1. try_files 配置错误2. root 路径错误解决方法1. 检查 try_f