C\C++
来源:互联网 发布:淘宝在线客服兼职 编辑:程序博客网 时间:2024/06/03 09:24
1.定义
MSDN上的解释为:
The sizeof keyword gives the amount of storage, in bytes, associated with a variable or a type (including aggregate types).
This keyword returns a value of type size_t.
其返回值类型为size_t,在头文件stddef.h中定义。这是一个依赖于编译系统的值,一般定义为:
2. 语法
sizeof有三种语法形式,如下:
2) sizeof( type_name ); // sizeof( 类型 );
3) sizeof object; // sizeof 对象;
sizeof( 2 ); // 2的类型为int,所以等价于 sizeof( int );sizeof( 2 + 3.14 ); // 3.14的类型为double,2也会被提升成double类型,所以等价于 sizeof( double );
char foo(){ printf("foo() has been called./n"); return 'a';}int main(){ size_t sz = sizeof( foo() ); // foo() 的返回值类型为char,所以sz = sizeof( char ),foo()并不会被调用 printf("sizeof( foo() ) = %d/n", sz); }
char* pc = "abc";int* pi;string* ps;char** ppc = &pc;void (*pf)(); // 函数指针sizeof( pc ); // 结果为4sizeof( pi ); // 结果为4sizeof( ps ); // 结果为4sizeof( ppc ); // 结果为4sizeof( pf ); // 结果为4
4. 数组的sizeof
数组的sizeof值等于数组所占用的内存字节数,如:
char a1[] = "abc";int a2[3];sizeof( a1 ); // 结果为4,字符串末尾还存在一个NULL终止符sizeof( a2 ); // 结果为3*4=12(依赖于int)
void foo3(char a3[3]){ int c3 = sizeof( a3 ); // c3 == 4}void foo4(char a4[]){ int c4 = sizeof( a4 ); // c4 == 4}
结构体的sizeof需要注意的一点是字节对齐
struct S1{ char c; int i;};sizeof(S1);// 8
struct S3{ char c1; S1 s; char c2;};sizeof(S3);//16
S1的最宽基本成员的类型为int,S3在考虑最宽基本类型成员时是将S1“打散”看的,所以S3的最宽基本类型为int,这样,通过S3定义的变量,其存储空间首地址需要被4整除,整个sizeof(S3)的值也应该被4整除。
空结构体的大小不为0,而是1
struct S5 { };sizeof( S5 ); // 结果为1
6. 联合体的sizeof
结构体在内存组织上是顺序式的,联合体则是重叠式,各成员共享一段内存,所以整个联合体的sizeof也就是每个成员sizeof的最大值。结构体的成员也可以是复合类型,这里,复合类型成员是被作为整体考虑的。
所以,下面例子中,U的sizeof值等于sizeof(s)。
union U{ int i; char c; S1 s;};
(1) 对于用sizeof()来测试类的大小,要注意一下几点,一就是空类的大小,当一个类中没有变量同时也没有虚函数时,其大小为1
class X{};X x;cout<<sizeof(x)<<endl;
class classA{char a;int b;double c;// int d;};
这里double最大,占八个字节, char 和 int 要和八个字节对齐, 两个加起来不到八个字节算成八个字节,所以sizeof(classA)=16, (如果不注释d, 两个int就是八个字节了,所以sizeof(classA)=24, char补上了七个字节,凑够八个,加上两个int和一个double).
class classB{char a;// double b;// int c;virtual func1(){} };
因为虚函数指针占了四个字节,char占了一个字节,安照对齐原则,sizeof(classB)=8, 如果去掉注释int c,那就变成了12. 但,注意如果现在注释int c,而去掉double b的注释,guess what's it? 按一般来说,应该是16,一个double加char和一个指针,补足16位,但实际上是24,因为指针是单独拿出来算了,这就变成了double,加上char补足的八位,加上指针补足的八位,一共是24个字节了.
class A2{ public: int a; private: char b;};class A3:public A2{ public: char b; short a; };sizeof(A3)是8. 但如果A3如下:class A3:public A2{ public: short a; char b; };sizeof(A3)是12.
9.子类和父类都有虚函数情况,子类的sizeof是它父类成员(无论成员是public或private),再加上它自己的成员,对齐后的sizeof,再加4(虚表指针)
class Base{ public: Base(){cout<<"Base-ctor"<<endl;} ~Base(){cout<<"Base-dtor"<<endl;} int a; virtual void f(int) {cout<<"Base::f(int)"<<endl;} virtual void f(double){cout<<"Base::f(double)"<<endl;}};class Derived:public Base{ public: Derived(){cout<<"Derived-ctor"<<endl;} int b; virtual void g(int){cout<<"Derived::g(int)"<<endl;}};sizeof(Derived)是12.
10.对于虚继承的子类,其sizeof的值是其父类成员,加上它自己的成员,以及它自己一个指向父类的指针,对齐后的sizeof
#include <iostream.h> class a { private: int x; }; class b: virtual public a { private: int y; }; class c: virtual public a { private: int z; }; class d:public b,public c { private: int m; }; int main(int argc, char* argv[]) { cout<<sizeof(a)<<endl; cout<<sizeof(b)<<endl; cout<<sizeof(c)<<endl; cout<<sizeof(d)<<endl; return 0; }
调试结果为
4
12
12
24
sizeof(b)和sizeof(c)相同,都是4+4+4=12。
sizeof(d)是sizeof(b)(为12)+sizeof(c)(为12)-b和c相同的部分(a的成员,大小是4)+d自己的成员(大小为4)=24
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;}};class Derived:virtual public Base{public: Derived(){cout<<"Derived-ctor"<<endl;} virtual void g(int){cout<<"Derived::g(int)"<<endl;}};
sizeof(Base)=4
sizeof(Derived)=12 (父类虚表指针大小4+自己虚表指针大小4+子类指向父类的一个指针大小4=12)
- c
- c
- c
- c
- C
- c
- c
- c
- C+
- c
- C
- c
- c
- c
- C
- C
- c
- C
- Html5支持iPhone触屏的脚本
- 设置root密码
- android mainfest总配置文件相关权限声明
- Git基础
- RTX开发反向登录错误 HRESULT:0xFFFFBA9E
- C\C++
- 织梦(dede)标签大全
- JAVA 注解(Annotation)
- 探讨下一代无线终端硬件设计与测试 加速4G产品上市
- 内存和CPU匹配方法详解
- C++中友元详解
- Launcher2 Wallpaper 背景设置模块
- java的System.getProperty()方法可以获取的值
- 在glassfish部署应用成功后,访问时出类转换异常的解决