sizeof操作的个人使用心得

     说起sizeof,想来大家都不陌生,她是用来计算数据所占内存空间字节数的一个操作符。

    但是,很多人,尤其是初学者,甚至我的同事中一些已经开发过一段时间的编程人员对于这个操作符的使用都存在不少理解上的失误。

    今天,我就简单分享一些我个人的使用心得,不是什么了不起的东西,毕竟本人也只能算是编程界的菜鸟一枚,但是希望能对初学者们有些帮助。

首先,对于做一个简单的sizeof常数汇总,如下:

    sizeof int:4
    sizeof short:2
    sizeof long:4
    sizeof float:4
    sizeof double:8
    sizeof char:1
    sizeof *p:4
    sizeof WORD:2
    sizeof DWORD:4

然后,关于sizeof有一个使用注意点:

    string str = "abcde";

    sizeof(str)得到的是string类的大小,用sizeof(str.c_str())或者sizeof(str.data())将string类型转换成char*才能获取到这个字符串的值。

下面说一说指针,关于指针有个经典实例,都是老生长谈啦:

经典问题: 
      double* (*a)[3][6];
 
      cout<<sizeof(a)<<endl; // 4 a为指针
      cout<<sizeof(*a)<<endl; // 72 *a为一个有3*6个指针元素的数组
      cout<<sizeof(**a)<<endl; // 24 **a为数组一维的6个指针
      cout<<sizeof(***a)<<endl; // 4 ***a为一维的第一个指针
      cout<<sizeof(****a)<<endl; // 8 ****a为一个double变量
问题解析: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

    这个是本人直接从网上拷贝的,实际上,这个我本人并不想多说,都是关于指针需要掌握的知识,也没人会这么用。不过关于指针,我个人倒是的确有一个注意点要提一提:

    比如说:char* p;sizeof(p)的结果是4,sizeof(*p)相当于sizeof(char)的结果是1;当然,这不是重点,我要说的重点在于数组。是的,数组才是我要提的注意点。

    通常情况下,如int a[16];char b[] = "abcde";等等来说,我们可以将这些数组看作是固定大小的指针来使用,但是,在使用sizeof(a)、sizeof(b)时,我们实际上取得的是这些“静态数组”的值,也就是说sizeof(a) = 16;sizeof(b) = 6;

   只有当数组作为形参,没有固定分配内存时,sizeof这个数组的值才会按照指针计算为4字节。

最后,我要讲一讲今天的重头戏,也是在使用sizeof最容易出错的地方,struct。

    我先不说其他,大家先看看以下的几个struct所占的字节数应该是多少。

struct A1
{
    char a;
    char b;
    int c;
    double d;
};
struct A2
{
    char a;
    double d;
    char b;
    int c;
};
struct A3
{
    char a[8];
    int b;
    short c;
    short d;
};
struct A4
{
    A1 a;
    int b;
};
struct A5
{
    A3 a;
    int b;
};

    各位,想好了没?

    想好了我可公布答案啦:

sizeof(A1) == 16;

sizeof(A2) == 24;

sizeof(A3) ==16;

sizeof(A4) == 24;

sizeof(A5) == 20;

    各位看客,能想清楚不?

这里我写一下我对在struct中使用sizeof的理解:在struct中用sizeof取值时,在未通过类似#pragma pack(1)来关闭内存对齐时,编译器在编译时,对于结构体,总会分配一个该结构体中占用字节数最多的类型所占用内存的整数倍的内存。而且,这种分配方式,我取了一个名字叫做“内存分割”。这么意思呢?

    以A1、A2为例,两者中的变量看似都是一样的,但是编译器按照顺序分配内存时,A1在分配char,char,int时,按照最长字节数8来分配并没有填充满,所以只占8个字节,加上double 占的8个字节一共16个字节。而对于A2而言,在分配到第二个double时,已经超出了8个字节,所以第一个char 占了8个字节,第二个double 占了8个字节,最后两个数占了8个字节,所以一共占了24个字节。

   而下面有同学就问了,为啥sizeof(A4)和sizeof(A5)的值不一样啊,而且为什么不是16的倍数呢?A1和A3的sizeof值不都是16嘛?

   这里就又是另一个坑了,编译器在做内存对齐的时候,如果struct内部有另一个struct时,她比较的是struct中对齐所用的那个值,所以A1对齐用的值是8个字节,A3对齐用的是4个字节,所以A4和A5的值就不一样了。

好了,以上就是本人对于sizeof的一些浅见,如果有什么不对的地方,希望各位大神指点。

posted @ 2014-12-23 22:36  SarielTang  阅读(160)  评论(0编辑  收藏  举报