sizeof终极解密

sizeof,在很多招聘笔试或者面试都会提到,它里面到底隐藏着多少陷阱呢?就让我们一探究竟。

铭记一个宗旨:sizeof计算的是一种数据类型(可以是结构体、int 、char、类等)所占的内存大小,只与类型有关,我们时刻关注的点应该是变量本身的类型,而不是它的值或指向的内存。

注:sizeof是运算符,不是函数。

一般可以这样理解:

数组——编译时分配的数组空间大小;
指针——存储该指针所用的空间大小(存储该指针的地址的长度,是长整型,应该为4);
对象——对象的实际占用空间大小;
函数——函数的返回类型所占的空间大小。函数的返回类型不能是void;

本质上都可以理解为:

“类型”——该“类型”所占的空间大小。

示例1:计算一般类型数组指针

 

#include<stdio.h>

int main()
{
    char arr[10]="What?";
    char *parr=arr;
    char *pchar=NULL;


    int a[10];
    int *pa=a;
    int *pint;

    printf("sizeof(arr)=%d\n",sizeof(arr));//10
    printf("sizeof(parr)=%d\n",sizeof(parr));//4
    printf("sizeof(*parr)=%d\n",sizeof(*parr));//1
    printf("sizeof(pchar)=%d\n",sizeof(pchar));//4
    printf("sizeof(*pchar)=%d\n\n",sizeof(*pchar));//1

    printf("sizeof(a)=%d\n",sizeof(a));//40
    printf("sizeof(pa)=%d\n",sizeof(pa));//4
    printf("sizeof(*pa)=%d\n",sizeof(*pa));//4
    printf("sizeof(pint)=%d\n",sizeof(pint));//4
    printf("sizeof(*pint)=%d\n\n",sizeof(*pint));//4

    return 0;
}


运行结果如下:

 

解析如下:

sizeof(arr)=10; //其计算的类型为 sizeof(char*[10])  char乘以10,所以为10;

sizeof(parr)=4;//其计算的类型为sizeof(char *) 是一个指针类型,指针的存储用的是长整型,所以为4;

sizeof(*parr)=1;//其计算的类型为sizeof(char) 是一个字符型,所以为1;

注:与其值是多少(或指针指向什么,如NULL)无关;

 

示例3:计算复合类型(结构体、枚举等)

#include<stdio.h>

//结构体
typedef struct TESTstruct
{
    char a;
    short b;
    int c;
}TESTstruct;

//枚举
typedef enum
{
    a=0,
    b,
    c
}TESTenum;

//共用体
typedef union
{
    char a;
    short b;
    int c;
}TESTunion;

int main()
{
    //把复合类型看成int等之类的普通类型对待即可
    TESTstruct tmpstruct1;
    TESTstruct *tmpstruct2;

    TESTenum tmpenum1;
    TESTenum *tmpenum2;

    TESTunion tmpunion1;
    TESTunion *tmpunion2;

    printf("sizeof(tmpstruct1)=%d\n",sizeof(tmpstruct1));//8
    printf("sizeof(tmpstruct2)=%d\n",sizeof(tmpstruct2));//4
    printf("sizeof(*tmpstruct2)=%d\n\n",sizeof(*tmpstruct2));//8

    //C99把枚举类型也作为整型数据(int)中的一种
    printf("sizeof(tmpenum1)=%d\n",sizeof(tmpenum1));//4
    printf("sizeof(tmpenum2)=%d\n",sizeof(tmpenum2));//4
    printf("sizeof(*tmpenum2)=%d\n\n",sizeof(*tmpenum2));//4

    printf("sizeof(tmpunion1)=%d\n",sizeof(tmpunion1));//4
    printf("sizeof(tmpunion2)=%d\n",sizeof(tmpunion2));//4
    printf("sizeof(*tmpunion2)=%d\n\n",sizeof(*tmpunion2));//4

    return 0;
}


运行结果如下:

 

注:

1、把复合类型看成int等之类的普通类型对待即可,解析过程同普通类型;

2、C99把枚举类型也作为整型数据(int)中的一种;

3、在联合体中取最大成员所占的内存;

 

示例4:对于动态内存

 

#include<stdio.h>
#include<malloc.h>

int main()
{
    void *p=malloc(sizeof(char)*10);
    void *pint=malloc(sizeof(int)*10);

    int *pint1=(int *)malloc(sizeof(int)*10);

    printf("sizeof(*p)=%d\n",sizeof(*p));//1
    printf("sizeof(p)=%d\n\n",sizeof(p));//4

    printf("sizeof(*pint)=%d\n",sizeof(*pint));//1
    printf("sizeof(pint)=%d\n\n",sizeof(pint));//4

    printf("sizeof(*pint1)=%d\n",sizeof(*pint1));//4
    printf("sizeof(pint1)=%d\n\n",sizeof(pint1));//4

    printf("sizeof(malloc(n))=%d\n",sizeof(malloc(100)));//4

    return 0;
}

运行结果如下:

 

解析:

1、malloc函数返回的void *,为一指针,因此sizeof(malloc(n))=4;

2、sizeof(*p)=1可以理解为malloc默认都是以单字节为基本单位申请内存,因此返回的从存储的角度看等同于char *p(实际上并非等同),根据示例1的分析可知为1;

3、sizeof(*pint1)=4此时的分析完全跟示例1一致。

 

类型5:针对C++中对象

还是得铭记宗旨,sizeof计算的是一种数据类型所占的内存大小,只与类型有关。因为对象是类的实例化,因此计算对象的大小实则计算C++中class类的这一数据类型(姑且这么认为)所占的内存大小。

示例如下:

 

#include<iostream>

using namespace std;

//空类
class A
{
};

class B
{
private:
    int b1;
};

class C
{
private:
    int c1;
    char c2;
};

class D
{
public:
     D(){}
     virtual ~D(){}
private:
     int d1;
     char *d2;
};

class E
{
public:
    E(){}
    virtual ~E(){}
private:
    int a ;
    char *p;
    static int b;
};

class F:public E
{
public:
    F(){}
    ~F(){}
private:
    int c;
};

class G: public virtual E
{
public:
    G(){}
    ~G(){}
private:
    int c;
};

class H: public E
{
public:
    H(){}
    ~H(){}
    virtual void GetValue(){}
private:
    int c;
};

class I: public virtual E
{
public:
    I(){}
    ~I(){}
    virtual void GetValue(){}
private:
    int c;
    int a;
};

int main()
{
    A a;
    B b;
    C c;
    D d;
    E e;
    F f;
    G g;
    H h;
    I i;

    cout<<"sizeof(A)="<<sizeof(A)<<endl;//1
    cout<<"sizeof(a)="<<sizeof(a)<<"\n"<<endl;

    cout<<"sizeof(B)="<<sizeof(B)<<endl;//4

    cout<<"sizeof(C)="<<sizeof(C)<<endl;//8

    cout<<"sizeof(D)="<<sizeof(D)<<endl;//12

    cout<<"sizeof(E)="<<sizeof(E)<<endl;//12

    cout<<"sizeof(F)="<<sizeof(F)<<endl;//12+4

    cout<<"sizeof(G)="<<sizeof(G)<<endl;//12+4+4

    cout<<"sizeof(H)="<<sizeof(H)<<endl;//12+4

    cout<<"sizeof(I)="<<sizeof(I)<<endl;//12+4+4+4
    return 0;
}

 

运行结果如下:

解析:

1、针对空类,编译器默认分配子一个字节;

2、计算类大小时符合内存对齐原则,可以类比于结构体;

3、当类含有虚函数时,那么类中就有一个被隐藏的成员变量信息:虚函数指针(4个字节),这个指针指向一个虚函数表,即使有多个虚函数, 虚函数表只有一个,因此只有一个虚函数指针;

4、当类中含有静态数据成员时,由于静态成员变量是在静态存储区分配空间的,它不属于类空间的一部分,因此类中的static成员变量不占据空间;

5、非虚继承派生类对象的存储空间 = 基类存储空间 + 派生类特有的非static数据成员的空间(如果派生类有虚函数且基类也有,则忽略派生类虚函数的计算);

6、如果是虚继承的话,类对象的存储空间大小 = 基类的存储空间 + 派生类特有的非static数据成员的存储空间 + 虚继承(+4)(注:此时忽略派生类中的虚函数);

 

习题解析:

typedef  union  

 {

long   i;   

int  k[5];   

char   c;

} DATE;//最大类型为int[5],所以sizeof(DATE)=20;

typedef struct  data   

{   

int   cat; //0-3

DATE   cow; //4-23  

double   dog;//24-31

} too; //因此sizeof(too)=32;

 

C99标准规定:void返回类型的函数(void *除外,返回的是指针)、位域(bit-field)成员不能被计算sizeof值

即:

void func1();

sizeof(func1());//错误

 

typedef struct tmp

{

unsigned int f1:1;

unsigned int f2:5;

unsigned int f3:12;

}tmp;

tmp a;

sizeof(a.f1);//错误

 

附:

char str[]="hello world";
sizeof(str)=12;

此处由于并未确定数组大小,因此我们可以理解为此处计算的实为实际数组的大小,因此为12(别忘了还有“\0”);

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值