【C++快速上手】六、sizeof学习笔记

本文深入探讨C++中类的内存布局原理,包括空类、虚函数、成员函数、静态数据成员、普通继承与虚继承对类大小的影响,以及字节对齐规则。通过具体代码示例,详细解释了各种情况下类的大小计算方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

总结

  • 原则1:空类的大小为1字节
  • 原则2:一个类中,虚函数本身、成员函数(包括静态与非静态)和静态数据成员都是不占用类对象的存储空间。
  • 原则3:对于包含虚函数的类,不管有多少个虚函数,只有一个虚指针vptr的大小。
  • 原则4:普通继承和派生类继承了所有基类的函数与成员,要按照字节对齐来计算大小
  • 原则5:虚函数继承,不管是单继承还是多继承,都是继承了基类的vptr。(32位操作系统4字节,64位操作系统 8字节)!
  • 原则6:虚继承,继承基类的vptr

原则1:空类的大小为1字节

#include<iostream>
using namespace std;
class A{};
int main()
{
    cout<<sizeof(A)<<endl;
    return 0;
}

//输出
1

原则2:一个类中,虚函数本身、成员函数(包括静态与非静态)和静态数据成员都是不占用类对象的存储空间。

#include<iostream>
using namespace std;
class A
{
    public:
        char b;
        virtual void fun() {};
        static int c;
        static int d;
        static int f;
};

int main()
{
    /**
     * @brief 16  字节对齐、静态变量不影响类的大小、vptr指针=8
     */
    cout<<sizeof(A)<<endl; 
    return 0;
}

//输出
16

上述例子中,只用虚指针占了八个字节(虚函数不占存储空间),非静态数据成员char b占了1个字节,又由于字节对齐,所以一共占了16个字节!

原则3:对于包含虚函数的类,不管有多少个虚函数,只有一个虚指针vptr的大小。

#include<iostream>
using namespace std;
class A{
    virtual void fun();
    virtual void fun1();
    virtual void fun2();
    virtual void fun3();
};
int main()
{
    cout<<sizeof(A)<<endl; // 8
    return 0;
}

//输出
8

原则4与5:

/**
 * @brief 1.普通单继承,继承就是基类+派生类自身的大小(注意字节对齐)
 * 注意:类的数据成员按其声明顺序加入内存,与访问权限无关,只看声明顺序。
 * 2.虚单继承,派生类继承基类vptr
 */

#include<iostream>

using namespace std;

class A
{
public:
    char a;
    int b;
};

class B:A
{
public:
    short a;
    long b;
};

class C
{
    A a;
    char c;
};

class A1
{
    virtual void fun(){}
};

class C1:public A
{
};

int main()
{
    cout<<"char:"<<sizeof(char)<<endl;//1
    cout<<"int:"<<sizeof(int)<<endl;//4
    cout<<"short:"<<sizeof(short)<<endl;//2
    cout<<"long:"<<sizeof(long)<<endl;//4

    cout<<"A:"<<sizeof(A)<<endl; // 8=4+4
    cout<<"B:"<<sizeof(B)<<endl; // 16=8+4+4
    cout<<"C:"<<sizeof(C)<<endl; // 12=8+4
    cout<<"A1:"<<sizeof(A1)<<endl;// 8
    cout<<"C1:"<<sizeof(C1)<<endl; // 8

    return 0;
}

//输出
char:1
int:4
short:2
long:4
A:8
B:16
C:12
A1:8
C1:8

原则6:虚继承,继承基类的vptr

#include<iostream>
using namespace std;
class A
{
    virtual void fun() {}
};
class B
{
    virtual void fun2() {}
};
class C : virtual public  A, virtual public B
{
public:
    virtual void fun3() {}
};

int main()
{
    /**
     * @brief 8 8 16  派生类虚继承多个虚函数,会继承所有虚函数的vptr
     */
    cout<<sizeof(A)<<" "<<sizeof(B)<<" "<<sizeof(C);

    return 0;
}

//输出
8 8 16

注意最后一个类C的输出,是16,fun3没有作用,也就是说:虚函数只继承基类的vptr!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

ReCclay

如果觉得不错,不妨请我喝杯咖啡

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值