sizeof

一、sizeof使用的场合:

1、sizeof操作符的一个主要用途是与存储分配和I/O系统那样的例程进行通信。例如:

1 void* malloc(size_t size);
2 size_t fread(void *ptr, size_t size, size_t nmemb, FILE* stream)

2、用它可以看看某种类型的对象在内存中所占的单元字节。例如:

1  void *memset(void *s, int c, sizeof(s));

3、在动态分配一对象时,可以让系统知道要分配多少内存。

4、便于一些类型的扩充。在windows中很多结构类型就有一个专用的字段用来存放该类型的字节大小。

5、由于操作数的字节数在实现时可能出现变化,建议在涉及操作数字节大小时用sizeof代替常量计算。

6、如果操作数就是函数中的数组形参或函数类型的形参,sizeof给出其指针的大小。

 

二、sizeof不能使用的场合:

sizeof操作符不能用于函数类型,不完全类型或位字段。不完全类型指具有未知存储大小的数据类型,如未知存储大小的数组类型、未知内容的结构或联合类型等。

 

三、

 1 cout<<sizeof(string*)<<endl; // 4
 2 cout<<sizeof(int*)<<endl; // 4
 3 cout<<sizof(char****)<<endl; // 4
 4 
 5 int f();
 6 cout << sizeof(f) << endl;  //编译错误
 7 cout << sizeof(f()) << endl; //4
 8 
 9 int a=6;
10 cout << sizeof(a=8) << endl;  //输出4,
11 cout << a << endl; //输出6。sizeof只是个运算符,所以相当于符号替换,a=8被替换成int,并不会修改a的值。

 

四、

 1 #include <iostream>
 2 using namespace std;
 3 
 4 int Sum(int i[]) {
 5     int sumofi = 0;
 6     for (int j = 0; j < sizeof(i)/sizeof(int); j++)   //实际上,sizeof(i) = 4这里数组名在形参作为指针
 7        sumofi += i[j];
 8     return sumofi;
 9 }
10 
11 int main() {
12     int allAges[6] = {21, 22, 22, 19, 34, 12};
13     cout<<Sum(allAges)<<endl;
14      return 0;
15 }
View Code
Sum的本意是用sizeof得到数组的大小,然后求和。但是实际上,传入自函数Sum的,只是一个int 类型的指针,所以sizeof(i)=4,而不是24,所以会产生错误的结果。解决这个问题的方法使是用指针或者引用。
一般求数组的长度我们会习惯用函数模板。这个可以参考文章《函数模板》。
 
五、
1 double* (*a)[3][6]; 
2 cout<<sizeof(a)<<endl; // 4 a为指针
3 cout<<sizeof(*a)<<endl; // 72 *a为一个有3*6个指针元素的数组
4 cout<<sizeof(**a)<<endl; // 24 **a为数组一维的6个指针
5 cout<<sizeof(***a)<<endl; // 4 ***a为一维的第一个指针
6 cout<<sizeof(****a)<<endl; // 8 ****a为一个double变量
View Code
解析:a是一个很奇怪的定义,他表示一个指向double*[3][6]类型数组的指针。既然是指针,所以sizeof(a)就是4 既然a是执行double*[3][6]类型的指针,*a就表示一个double*[3][6]的多维数组类型,因此sizeof(*a)=3*6*sizeof(double*)=72。同样的,**a表示一个double*[6]类型的数组,所以sizeof(**a)=6*sizeof  (double*)=24***a就表示其中的一个元素,也就是double*了,所以sizeof(***a)=4。至于****a,就是一个double了,所以sizeof(****a)=sizeof(double)=8。 
 1 #include <iostream>
 2 
 3 using namespace std;
 4 
 5 int main() {
 6    char (*f)[3];
 7    int **c[3][4];
 8    double *(*a)[3][6];
 9    cout << sizeof(f) << endl;   //4
10    cout << sizeof(*f) << endl;  //3
11    cout << sizeof(**f) << endl;  //1
12    cout << sizeof(a) << endl;  //4
13    cout << sizeof(*a) << endl;  //72
14    cout << sizeof(**a) << endl;  //24
15    cout << sizeof(***a) << endl;  //4
16    cout << sizeof(****a) << endl;  //8
17    cout << sizeof(c) << endl;    //48
18     return 0;
19 }
View Code
六、sizeof(class)
空类的大小为1,其他跟struct差不多,可以参见《字节对齐详解》里面的内容。需要指出的是class里面的virtual函数是独立开来的,virtual代表的是一个虚函数指针,它是一个虚函数表的地址入口。例如:
1 class classB {
2     char a;
3     // double b;
4     // int c;
5     virtual func1(){} 
6 };
因为虚函数指针占了四个字节,char占了一个字节,安照对齐原则,sizeof(classB)=8, 如果去掉注释int c,那就变成了12. 但,注意如果现在注释int c,而去掉double b的注释,guess what's it? 按一般来说,应该是16,一个double加char和一个指针,补足16位,但实际上是24,因为指针是单独拿出来算了,这就变成了double,加上char补足的八位,加上指针补足的八位,一共是24个字节了。
 

 

posted @ 2013-10-30 00:45  RunningPower  阅读(229)  评论(2编辑  收藏  举报