`
kmplayer
  • 浏览: 512130 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

存在虚基类时,类对象的大小

阅读更多
1,
实例代码:
#include <iostream>
using namespace std;

class Top
{
protected:
    int x;
public:
    Top(int n)
    {
        x = n;
    }
    virtual ~Top() {}
};

class Left : virtual public Top
{
protected:
    int y;
public:
    Left(int m, int n) : Top(m)
    {
        y = n;
    }
};

class Right : virtual public Top
{
protected:
    int z;
public:
    Right(int m, int n) : Top(m)
    {
        z = n;
    }
};

class Bottom : public Left, public Right
{
    int w;
public:
    Bottom(int i, int j, int k, int m) : Top(i), Left(0, j), Right(0, k)
    {
        w = m;
    }
};

int main()
{
  Bottom b(1, 2, 3, 4);

  cout << sizeof(Top) << endl;
  //Top: int+虚表指针  8
  cout << sizeof(Left) << endl;
  //Left: 大小为Top+int+指向虚基类的指针  16
  cout << sizeof(Bottom) << endl;
  //Bottom: 大小为Top+Left(int + 指向虚基类的指针)+Right(int + 指向虚基类的指针)+int 28
}

(2)一个比较奇怪的地方
#include <iostream>
using namespace std;

class Top
{
public:
    Top()
    {
    }
    virtual ~Top() {}
private:
    int i;
};

class Left : virtual public Top
{
public:
    Left()
    {
    }
};

class Right : virtual public Top
{
public:
    Right()
    {
    }
};

class Bottom : public Left, Right
{
};


class Top2
{
public:
    Top2()
    {
    }
    virtual ~Top2() {}
};

class Left2 : virtual public Top2
{
public:
    Left2()
    {
    }
};

class Right2 : virtual public Top2
{
public:
    Right2()
    {
    }
};

class Bottom2 : public Left2, Right2
{
};

int main()
{
    cout << sizeof(Top) << endl;
    cout << sizeof(Left) << endl;
    //Left: Top+虚表指针  12
    cout << sizeof(Bottom) << endl;
    //Left: Top+虚表指针+虚表指针  16

    cout << sizeof(Top2) << endl;
    cout << sizeof(Left2) << endl;
    //Left2: 为什么是4?
    cout << sizeof(Bottom2) << endl;
    //为什么 是 8?
}

(3) 多重虚继承的情况
#include <iostream>
using namespace std;

class A
{
public:
     virtual void aa()
     {
     }
private:
     char k[3];
};

class B : virtual public A
{
public:
     virtual void bb()
     {
     }
private:
     char j[3];
};

class C : virtual public B
{
public:
     virtual void cc()
     {
     }
private:
     char i[3];
};

class D : public B
{
public:
     virtual void cc()
     {
     }
private:
     char i[3];
};

int main()
{
     A a;
     B b;
     C c;
     cout<<"sizeof(A):"<<sizeof(A)<<endl;
     cout<<"sizeof(B):"<<sizeof(B)<<endl;
     //A+4+虚表指针 16
     cout<<"sizeof(C):"<<sizeof(C)<<endl;
     //B+4+续表指针 24
     cout<<"sizeof(D):"<<sizeof(D)<<endl;
     //B+4 20
     return 0;
}
分享到:
评论

相关推荐

    C++经典资料\虚基类与虚函数

    此外,虚基类还允许我们使用基类的指针或引用直接指向派生类的对象,无需进行类型转换。例如,`C obj; L * ptr = &obj;`是合法的,这意味着可以通过虚基类指针访问派生类的所有公共基类成员。然而,虚基类的指针或...

    C++多继承与虚基类

    虚基类的主要目的是确保只有一个实例存在,即使多个派生类通过不同的路径继承同一个基类。在声明基类时,通过在基类名前加上`virtual`关键字,可以将其设置为虚基类。这样,当一个类多继承时,所有从这个虚基类派生...

    3.1C++类对象的大小共2页.pdf.zip

    在多继承的情况下,类的对象大小可能会因为基类的贡献而增大。如果一个类从多个基类派生,且这些基类都有非空的数据成员或虚函数,那么这些基类的大小都会累加到派生类对象的大小中。需要注意的是,由于二进制布局的...

    C++中虚函数工作原理和(虚)继承类的内存占用大小计算1

    此外,虚继承的派生类在构造时需要特殊的初始化步骤,确保虚指针正确地初始化到共享基类的虚函数表。 总结来说,C++的虚函数和虚继承是实现多态性和解决多重继承问题的关键机制,它们带来了一些额外的内存开销和...

    图说C++对象模型.pdf

    在多重继承中,如果使用了虚继承,子类的对象内存布局会包含一个虚基类表指针(vbptr),该指针指向一个虚基类表(vbtable),这个表记录了与虚基类相关的偏移量信息。虚拟菱形继承是多重虚继承中的一种特例,它需要...

    深入探索C++对象模型

    这在继承体系中尤其重要,因为一个基类可能被多个派生类继承,而虚基类则确保了单一实例的存在。 C++对象模型可以分为几种不同的实现方式: 1. 简单对象模型:每个对象是一系列槽(slots),每个槽对应一个成员。...

    C++类的sizeof1

    总之,`sizeof`运算符在处理C++类时需要考虑类的成员大小、虚函数的存在以及继承关系等因素。理解这些细节对于优化内存使用和编写高效代码至关重要。在实际编程中,应时刻注意类的大小,特别是当内存管理成为一个...

    《面向对象技术》期末复习资料.doc

    - 虚基类 -&gt; 基类对象 -&gt; 成员对象 -&gt; 派生类自身 复习时,应重点掌握上述知识点,特别是类、对象、指针、引用、构造与析构函数、继承、多态以及静态成员的使用。通过理解这些概念,你将能够更好地理解和编写面向...

    谈VC++对象模型

    - 多重继承:如果类从多个基类派生,编译器需要解决二义性,可能会使用虚基类来确保唯一实例。 - 虚继承:为了解决多继承中基类的多次实例化问题,引入虚基类,增加额外的指针(vptr)以指向虚基类的单一实例。 2...

    华为笔试内部资料

    在C++中,若子类的析构函数为虚函数,则当通过基类指针删除派生类对象时,会首先调用派生类的析构函数,然后调用基类的析构函数。这是因为虚析构函数确保了对象层次结构中的析构函数能按照正确的顺序被调用,避免了...

    探索MinGW是如何存储对象、如何实现类继承、多态

    - 多继承时,情况更复杂,每个基类可能会有自己的vptr,派生类需要管理多个基类的虚函数表。 4. **C++类继承存储分配**: - 单继承时,子类对象可以直接转换为父类对象,反之则不允许,因为父类无法了解子类的...

    c++继承中的内存布局

    虚继承会引入一个额外的虚基类指针,以确保每个派生类都只拥有一份虚基类的实例。这会增加一定的内存开销。 - **虚函数调用**:虚函数调用涉及到虚函数表的查找,这意味着每次调用都需要一次间接跳转。此外,还需要...

    《Inside the C++ Object Model》笔记(1~7章)

    2. **多继承的基类表模型**:为了解决多继承带来的问题,每个类对象都指向一个基类表,该表包含了指向基类的指针,从而避免了派生类对象大小受多个基类的影响。 3. **C++的多继承内存模型**:C++采用了一种更复杂的...

    关于C++对象模型 --- by yangmeng

    - 强制转换到基类或虚基类涉及类型检查和指针调整,以确保正确访问对象的成员。 - 动态类型转换(如`dynamic_cast`)可以用于运行时类型检查和转换,但也有一定开销。 7. **异常处理**: - 异常处理允许程序在...

    sizeof_virtual.rar_测试函数_虚函数

    当一个指针或引用指向基类对象,但实际调用的是派生类的成员函数,这种机制就发挥了作用。为了支持这种行为,编译器会在基类中添加一个虚函数表(Virtual Table, VTT)的指针,这个指针在每个实例化对象中都存在。虚...

    c++练习题带答案57936.docx

    15. 虚基类的目的是解决多继承带来的二义性问题,它会在构建派生类对象时仅调用一次构造函数。声明`class B: virtual public A`表明类B继承自A,并且A是虚基类。 16. 在类A的构造函数中,需要分配足够大小的内存来...

    C++对象模型

    在对象模型中,子类的对象会包含基类的全部成员,这在内存中表现为子类对象的大小至少等于基类对象的大小加上子类自己添加的成员。 4. **虚函数表**:当涉及多态性时,C++使用虚函数表(VMT)来实现。每个具有虚...

    5 深度探索C++对象模型.rar

    在内存中,派生类对象的前部分是基类对象,接着是派生类特有的数据成员。如果基类有虚函数,派生类对象也会包含一个虚函数表指针。 - **多态性**:C++的多态主要通过虚函数实现。虚函数表使得在运行时能够动态绑定...

Global site tag (gtag.js) - Google Analytics