1.  定义:

    sizeof作用就是返回一个对象或者类型所占的内存字节数。

     它不是一个函数,其字节数的计算在程序编译时进行的。

2.  语法格式:

 (1)用于数据类型,使用形式: sizeof(type)

       数据类型必须用括号括住。

             sizeof(int)
 (2)用于变量,使用形式: sizeof(varname) 或 sizeof varname

            一般采用带括号的方式。

      int i; sizeof(i);

3.  返回结果:

    sizeof int:4
    sizeof short:2
    sizeof long:4
    sizeof float:4
    sizeof double:8
    sizeof char:1
    /*
       空类所占空间为1,单一继承、多重继承的空类所占空间也是1,虚表(虚指针)所占空间是4.
    */

(1)作用于指针:

        在32位计算机中,一个指针变量的返回值必定是4(32/8),但是,在64位系统中指针变量的sizeof结果为8。

(2)作用于数组:

    数组的sizeof值等于数组所占用的内存字节数,如:

       int a[5];  sizeof(a) = 20;
       char b[] = "abc"; sizeof(b)=4;//末尾存在一个终止符
       char str[20]="0123456789"; //strlen(str)=10(不含终止符);sizeof(str)=20;

(3)作用于函数参数:

    void func(char a[3])    
    {
        int c=sizeof(a);//c=4
    }

    这里函数参数a已不再是数组类型,而是变成了指针,相当于char* a,我们调用函数func时,程序不会在栈上分配一个大小为3的数组,数组是“传址”的,调用者只需将实参的地址传递过去,所以a自然为指针类型(char*),c的值也就为4。

(4)作用于联合体:

    结构体在内存组织上是顺序式的,联合体则是重叠式,各成员共享一段内存,所以整个联合体的sizeof也就是每个成员sizeof的最大值。

    union  u{             
            char a;
            int b;
            double c;
    }; //sizeof(u)=8

(5)作用于结构体:

遵循两个原则:
   1).整体空间是 占用空间最大的成员(的类型)所占字节数的整倍数
   2)数据对齐原则----内存按结构成员的先后顺序排列,当排到该成员变量时,其前面已摆放的空间大小必须是该成员类型大小的整倍数,如果不够则补齐,以此向后类推。

struct s1
{
char a;
double b;
int c;
char d; 
}; //sizeof(s1)=24; 各元素起始地址1-8-16-20,结构体在地址21处结束,大小需要是8的倍数24

struct s2
{
char a;
char b;
int c;
double d;
};//sizeof(s2)=16 ;各元素起始地址 1-2-4-8;结构体在15处结束,大小需要是8的倍数16

在自己定义结构体的时候,如果空间紧张的话,最好考虑对齐因素来排列结构体里的元素。

4. Sizeof与Strlen的区别与联系 

1)sizeof操作符的结果类型是size_t,它在头文件中typedef为unsigned int类型。其值在编译时即计算好了,strlen的结果要在运行的时候才能计算出来,是用来计算字符串的长度。

(2) sizeof是运算符,strlen是函数。 

(3)sizeof可以用类型做参数,strlen只能用char*做参数,且必须是以“\0”结尾的。

(4)数组做sizeof的参数不退化,传递给strlen就退化为指针了。 

注意:

    例1: strlen(char*)函数求的是字符串的实际长度,到遇到第一个'\0'结束,如果你只定义没有给它赋初值,这个结果是不定的,它会从aa首地址一直找下去,直到遇到'\0'停止

    char aa[10];cout<<strlen(aa)<<endl; //结果是不定的
    char aa[10]={'\0'}; cout<<strlen(aa)<<endl; //结果为0
    char aa[10]="jun"; cout<<strlen(aa)<<endl; //结果为3

    而sizeof返回的是变量声明后所占的内存数,不是实际长度。

    例2:

        char* ss = "0123456789"; 
        sizeof(ss); //结果 4 ;
        sizeof(*ss); //结果 1;*ss其实是获得了字符串的第一位'0'所占的内存空间;
        strlen(ss);//结果是10 ;