为什么这个程序中sizeof的结果是4…
2013-08-09 17:33
183 查看
为什么这个程序中sizeof的结果是4?
#include<iostream>
#include<complex>
using namespace std;
class base
{
public:
base(){cout<<"base-ctor"<<endl;}
~base(){cout<<"base-dtor"<<endl;}
virtual
void
f(int){cout<<"base::f(int)"<<endl;}
virtual
void
f(double){cout<<"base::f(double)"<<endl;}
virtual
void g(int
i=10){cout<<"base::g()"<<i<<endl;}
void
g2(int
i=10){cout<<"base::g2"<<i<<endl;}
};
class derived:public base
{
public:
derived(){cout<<"derived-ctor"<<endl;}
~derived(){cout<<"derived-dtor"<<endl;}
void
f(complex<double>){cout<<"derived::f(complex)"<<endl;}
virtual
void g(int
i=20){cout<<"derived::g()"<<i<<endl;}
};
void main()
{
base
b;
derived
d;
base
*pb=new derived;
cout<<sizeof(base)<<"tt"<<endl;
cout<<sizeof(derived)<<"bb"<<endl;
}
请看下面的定义(主要是定议的类"大"了点,我来个"小"点的)
class A
{
virtul
~A(){}
}
sizeof(A)=4 ? 你问的实际就是这个问题.
不知道你了不了解C++多态的实现方式?看没有看过一本书<<Inside
the C++ Object Model>>里面讲的很清楚.
当一个类中有虚函数存在时,编译器为自动为我个类加上一个成员(对于程序员来说,是透明的)vptr,也就是虚表指针.它指向一个虚表,表中存放的时类的虚函数地址,用于实现C++的动态绑定.
C++多态中的VPTR
以下文字摘录自<<C++编程思想>>(Bruce
Eckel):
C++中的关键字virtual告诉编译器对于某个成员函数进行动态绑定,而且自动装载实现动态绑定所必须的所有机制。
为了完成这件事,编译器对每个包含虚函数的类创建一个表
(称为VTABLE)。在VTABLE中,编译器旋转特定类的虚函数地址。在每个带有虚函数的类中,编译器“秘密”地置一指针,称为vpointer
(缩写为VPTR),指向这个对象的VTABLE。通过基类指针(或者引用)做虚函数调用时,也就是做多态调用时,编译器静态地插入取得这个VPTR,并在VTABLE表中查找函数地址的代码,这样就能调用正确的函数使动态绑定发生。
为了看到VPTR,特编写如下函数:
class no_virtual{
int a;
public:
void x() const {}
int i() const { return 1; }
};
class one_virtual{
int a;
public:
virtual void x() const {}
int i() const { return 1; }
};
class two_virtuals{
int a;
public:
virtual void x() const {}
virtual int i() const { return 1; }
};
void main()
{
cout<<"int:"<<sizeof(int)<<endl;
cout<<"no_virtual:"
<<sizeof(no_virtual)<<endl;
cout<<"void*:"<<sizeof(void*)<<endl;
cout<<"one_virtual:"
<<sizeof(one_virtual)<<endl;
cout<<"two_virtuals:"
<<sizeof(two_virtuals)<<endl;
}
不带虚函数,对象的长度恰好就是所期望的:单个int的长度。而带有单个虚函数的one_virtual,对象的长度是no_virtual的长度加上一个void指针的长度。它反映出,如果有一个或多个虚函数,编译器将在这个结构中插入一个指针(VPTR)。在one_virtual和two_virtuals之间没有区别。这是因为VPTR指向一个存放地址的表,只需要一个指针,因为所有虚函数地址都包含在这个表中。
#include<iostream>
#include<complex>
using namespace std;
class base
{
public:
base(){cout<<"base-ctor"<<endl;}
~base(){cout<<"base-dtor"<<endl;}
virtual
void
f(int){cout<<"base::f(int)"<<endl;}
virtual
void
f(double){cout<<"base::f(double)"<<endl;}
virtual
void g(int
i=10){cout<<"base::g()"<<i<<endl;}
void
g2(int
i=10){cout<<"base::g2"<<i<<endl;}
};
class derived:public base
{
public:
derived(){cout<<"derived-ctor"<<endl;}
~derived(){cout<<"derived-dtor"<<endl;}
void
f(complex<double>){cout<<"derived::f(complex)"<<endl;}
virtual
void g(int
i=20){cout<<"derived::g()"<<i<<endl;}
};
void main()
{
base
b;
derived
d;
base
*pb=new derived;
cout<<sizeof(base)<<"tt"<<endl;
cout<<sizeof(derived)<<"bb"<<endl;
}
请看下面的定义(主要是定议的类"大"了点,我来个"小"点的)
class A
{
virtul
~A(){}
}
sizeof(A)=4 ? 你问的实际就是这个问题.
不知道你了不了解C++多态的实现方式?看没有看过一本书<<Inside
the C++ Object Model>>里面讲的很清楚.
当一个类中有虚函数存在时,编译器为自动为我个类加上一个成员(对于程序员来说,是透明的)vptr,也就是虚表指针.它指向一个虚表,表中存放的时类的虚函数地址,用于实现C++的动态绑定.
C++多态中的VPTR
以下文字摘录自<<C++编程思想>>(Bruce
Eckel):
C++中的关键字virtual告诉编译器对于某个成员函数进行动态绑定,而且自动装载实现动态绑定所必须的所有机制。
为了完成这件事,编译器对每个包含虚函数的类创建一个表
(称为VTABLE)。在VTABLE中,编译器旋转特定类的虚函数地址。在每个带有虚函数的类中,编译器“秘密”地置一指针,称为vpointer
(缩写为VPTR),指向这个对象的VTABLE。通过基类指针(或者引用)做虚函数调用时,也就是做多态调用时,编译器静态地插入取得这个VPTR,并在VTABLE表中查找函数地址的代码,这样就能调用正确的函数使动态绑定发生。
为了看到VPTR,特编写如下函数:
class no_virtual{
int a;
public:
void x() const {}
int i() const { return 1; }
};
class one_virtual{
int a;
public:
virtual void x() const {}
int i() const { return 1; }
};
class two_virtuals{
int a;
public:
virtual void x() const {}
virtual int i() const { return 1; }
};
void main()
{
cout<<"int:"<<sizeof(int)<<endl;
cout<<"no_virtual:"
<<sizeof(no_virtual)<<endl;
cout<<"void*:"<<sizeof(void*)<<endl;
cout<<"one_virtual:"
<<sizeof(one_virtual)<<endl;
cout<<"two_virtuals:"
<<sizeof(two_virtuals)<<endl;
}
不带虚函数,对象的长度恰好就是所期望的:单个int的长度。而带有单个虚函数的one_virtual,对象的长度是no_virtual的长度加上一个void指针的长度。它反映出,如果有一个或多个虚函数,编译器将在这个结构中插入一个指针(VPTR)。在one_virtual和two_virtuals之间没有区别。这是因为VPTR指向一个存放地址的表,只需要一个指针,因为所有虚函数地址都包含在这个表中。
相关文章推荐
- 这个程序的结果为什么会这样 ?
- xcode6.1新建一个ios程序,为什么自动运行viewController这个类
- 编写一个java程序,从1加到100,结果是5050。 每做一次加法,就休眠1秒 当程序运行过程中,杀死这个程序。 再次运行,还能从上次的计算结果继续下去。
- VC++2005运行Win32控制台程序时,为什么弹不出控制台窗口(无法看到运行结果)?
- 在VS2005下用C++写的程序,安装到另一台未安装VS2005的电脑上,结果出现如下的问题:““由于应用程序的配置不正确,应用程序未能启动,重新安装应用程序可能会纠正这个问题”。在我自己的机子上就能正常安装运行。查了一下网上的方法: 方法一: 在类似
- 请帮忙看看这个程序的运行结果为何这样怪
- 为什么一个程序在vs2005中,按F5运行,和直接运行exe文件结果不一样
- linux引导程序为什么要把自己加载到内存的7c00h这个特定的地方?
- 帮小弟计算以下这个程序运行的结果
- [求助] 这个简单的程序为什么没有预期效果呢?
- 为什么sizeof一个empty class的结果是1
- 请帮忙看看这个程序的运行结果为何这样怪
- 请问这个程序为什么不能显示最后一行数据
- 给定一个正整数,编写程序计算有多少对质数的和等于输入的这个正整数,并输出结果。
- 大家看看下面这个程序会有什么结果
- VC++2005运行Win32控制台程序时,为什么弹不出控制台窗口(无法看到运行结果)?
- 帮忙看一下这个程序,为什么在调式时候调不出来
- 指针方面的问题,此程序调试显示是正确的,但是运行起来的结果是错误的为什么?求解
- 编写一个java程序,从1加到100,每做一次加法,就休眠1秒,当程序运行过程中,杀死这个程序。 再次运行,还能从上次的计算结果继续下去
- 这个Java程序运行不出来结果?