虚函数表指针、虚基类表指针
2015-09-16 20:30
162 查看
先看下面的代码:
class Base1
{
int a;
public:
virtual void fun1()
{
cout << "Base1::fun1" << endl;
}
};
class Base2
{
int b;
public:
virtual void fun1()
{
cout << "Base2::fun1" << endl;
}
virtual void fun2()
{
cout << "Base2::fun2" << endl;
}
};
class Derive1 : public Base1
{
int d1;
public:
virtual void fun1()
{
cout << "Derive1::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive1::d_fun1" << endl;
}
};
class Derive2 : virtual public Base1
{
int d2;
public:
virtual void fun1()
{
cout << "Derive2::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive2::d_fun1" << endl;
}
};
class Derive3 : virtual public Base2
{
int d3;
public:
virtual void fun1()
{
cout << "Derive3::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive3::d_fun1" << endl;
}
};
class Derive4 : public Derive2, public Derive3
{
int d4;
public:
virtual void fun1()
{
cout << "Derive4::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive4::d_fun1" << endl;
}
};
class Derive5 : public Base1, public Base2
{
virtual void d_fun()
{
cout << "Derive5::d_fun" << endl;
}
};
输出结果为:pstr1为虚函数表指针,vpstr为虚基类表指针
sizeof(Base1):8 //sizeof(int) + sizeof(pstr) = 4+4
[b]sizeof(Base12):8 //同上
[/b]
[b]sizeof(Derive1):12
//[b]sizeof(int) + sizeof(pstr) + sizeof(int)= 4+4+4[/b]
[/b]
[b]sizeof(Derive2):20
//[b]sizeof(int) + sizeof(Base1的pstr) +sizeof(Derive2的pstr) + sizeof(vpstr)+ sizeof(int)= 4+4+4+4+4[/b]
[/b]
[b][b]sizeof(Derive3):20
//同上
[/b][/b]
[b][b]sizeof(Derive4):44
//sizeof(Derive2) + sizeof(Derive3) + sizeof(int)
[/b][/b]
[b][b]sizeof(Derive5):16
//sizeof(Base1) + sizeof(Base2)
[/b][/b]
以下部分转自:http://blog.csdn.net/gukesdo/article/details/7593754
虚函数表指针、虚基类表指针
对于虚函数表指针和虚基类表指针:
当单继承且非虚继承时:每个含有虚函数的表只有一个虚函数表,所以只需要一个虚表指针即可;
当多继承且非虚继承时:一个子类有几个父类则会有几个虚函数表,所以就有和父类个数相同的虚表指针来标识;
总之,当时非虚继承时,不需要额外增加虚函数表指针。
当虚继承时:无论是单虚继承还是多虚继承,需要有一个虚基类表来记录虚继承关系,所以此时子类需要多一个虚基类表指针;而且只需要一个即可。
当虚继承时可能出现一个类中持有多个虚函数表的情况:无论是单虚继承还是多虚继承,
如果子类没有构造函数和析构函数,且子类中的虚函数都是在父类中出现的虚函数,这个时候不需要增加任何虚表指针;只需要像多继承那个持有父类个数的虚表指针来标识即可;
如果子类中含有构造函数或者析构函数或二者都有,则在子类中只要每出现一个父类中的虚函数则需要增加一个虚函数表指针来标识此类的虚函数表;
无论是否含有构造函数或者虚构函数,只要继承都是虚继承且出现了父类中没有出现的虚函数,则在子类中需要再增加一个徐函数表指针;如果其中有一个是非虚继承,则按照最省空间的原则,不需要增加虚函数表指针,因为这个时候可以和非虚基类共享一个虚函数表指针。
class Base1
{
int a;
public:
virtual void fun1()
{
cout << "Base1::fun1" << endl;
}
};
class Base2
{
int b;
public:
virtual void fun1()
{
cout << "Base2::fun1" << endl;
}
virtual void fun2()
{
cout << "Base2::fun2" << endl;
}
};
class Derive1 : public Base1
{
int d1;
public:
virtual void fun1()
{
cout << "Derive1::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive1::d_fun1" << endl;
}
};
class Derive2 : virtual public Base1
{
int d2;
public:
virtual void fun1()
{
cout << "Derive2::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive2::d_fun1" << endl;
}
};
class Derive3 : virtual public Base2
{
int d3;
public:
virtual void fun1()
{
cout << "Derive3::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive3::d_fun1" << endl;
}
};
class Derive4 : public Derive2, public Derive3
{
int d4;
public:
virtual void fun1()
{
cout << "Derive4::fun1" << endl;
}
virtual void d_fun1()
{
cout << "Derive4::d_fun1" << endl;
}
};
class Derive5 : public Base1, public Base2
{
virtual void d_fun()
{
cout << "Derive5::d_fun" << endl;
}
};
输出结果为:pstr1为虚函数表指针,vpstr为虚基类表指针
sizeof(Base1):8 //sizeof(int) + sizeof(pstr) = 4+4
[b]sizeof(Base12):8 //同上
[/b]
[b]sizeof(Derive1):12
//[b]sizeof(int) + sizeof(pstr) + sizeof(int)= 4+4+4[/b]
[/b]
[b]sizeof(Derive2):20
//[b]sizeof(int) + sizeof(Base1的pstr) +sizeof(Derive2的pstr) + sizeof(vpstr)+ sizeof(int)= 4+4+4+4+4[/b]
[/b]
[b][b]sizeof(Derive3):20
//同上
[/b][/b]
[b][b]sizeof(Derive4):44
//sizeof(Derive2) + sizeof(Derive3) + sizeof(int)
[/b][/b]
[b][b]sizeof(Derive5):16
//sizeof(Base1) + sizeof(Base2)
[/b][/b]
以下部分转自:http://blog.csdn.net/gukesdo/article/details/7593754
虚函数表指针、虚基类表指针
对于虚函数表指针和虚基类表指针:
当单继承且非虚继承时:每个含有虚函数的表只有一个虚函数表,所以只需要一个虚表指针即可;
当多继承且非虚继承时:一个子类有几个父类则会有几个虚函数表,所以就有和父类个数相同的虚表指针来标识;
总之,当时非虚继承时,不需要额外增加虚函数表指针。
当虚继承时:无论是单虚继承还是多虚继承,需要有一个虚基类表来记录虚继承关系,所以此时子类需要多一个虚基类表指针;而且只需要一个即可。
当虚继承时可能出现一个类中持有多个虚函数表的情况:无论是单虚继承还是多虚继承,
如果子类没有构造函数和析构函数,且子类中的虚函数都是在父类中出现的虚函数,这个时候不需要增加任何虚表指针;只需要像多继承那个持有父类个数的虚表指针来标识即可;
如果子类中含有构造函数或者析构函数或二者都有,则在子类中只要每出现一个父类中的虚函数则需要增加一个虚函数表指针来标识此类的虚函数表;
无论是否含有构造函数或者虚构函数,只要继承都是虚继承且出现了父类中没有出现的虚函数,则在子类中需要再增加一个徐函数表指针;如果其中有一个是非虚继承,则按照最省空间的原则,不需要增加虚函数表指针,因为这个时候可以和非虚基类共享一个虚函数表指针。
相关文章推荐
- linux之nm命令
- 【js】javascript变量声明优先级
- Java,JSP,JavaScript三和差异
- codeforces 577
- UIView 总复习
- 设计模式 工厂模式
- @ 小浩
- 猫猫学iOS 之微博项目实战(10)微博cell中图片的显示以及各种填充模式简介
- Java库使用----xstream1.3.1
- android学习:android Activity生命周期的七个方法测试
- 如何撰写商业计划书(精简)
- socket编程原理
- WebService之Axis2系列教程Axis2与Spring集成发布
- USACO Section 1.4 More Search Technique
- Radar Installation (贪心+数学 不错的题 )
- Count and Say
- 短信发送
- 设计模式 装饰者模式
- [家里蹲大学数学杂志]第418期南开大学2013年实变函数期末考试试题参考解答
- Very simple problem - SGU 111(大数开方)