听到过一个有趣且完美的解释——“sizeof”这个操作符可以类比为锤子,在锤子眼中,什么都是钉子,即万物皆可“敲”——求大小。
sizeof关键字给出了与变量或类型(包括聚合类型)相关联的存储容量(以字节为单位)。这个关键字返回size_t类型的值。
下面我们来看下几个实例:
int main()
{
int a[] = { 1,2,3,4 };
0 1 2 3
int (*p)[4] = &a;
printf("%d\n", sizeof(a));//4*4 = 16
printf("%d\n", sizeof(a + 0));//4/8 a+0是数组第一个元素的地址,是地址,大小就是4/8个字节
仅仅在sizeof(数组名)的时候表示的为整个数组,在数组名加数字后表示(首元素地址+数字)后的地址
printf("%d\n", sizeof(*a)); //4 a表示数组首元素的地址,*a表示数组的第一个元素,sizeof(*a)就是第一个元素的大小-4
printf("%d\n", sizeof(a + 1));//4/8 a表示数组首元素的地址,a+1数组第二个元素的地址,sizeof(a+1)就是第二个元素的地址的大小
printf("%d\n", sizeof(a[1]));//4 计算的是第二个元素的大小
printf("%d\n", sizeof(&a));//4/8 &a取出的是数组的地址,数组的地址也是地址呀,是地址大小就是4/8字节
printf("%d\n", sizeof(*&a));//16 计算的整个数组的大小
printf("%d\n", sizeof(&a + 1));//4/8 - &a是数组的地址,+1跳过整个数组,产生的4后边位置的地址
printf("%d\n", sizeof(&a[0]));//4/8 取出的数组第一个元素的地址
printf("%d\n", sizeof(&a[0] + 1));//4/8 数组第二个元素的地址
}
1.这里我们可以看到,在sizeof遇到数组名称的时候,读取的为整个数组大小,int类型下,读取的也就是4*n个字节长度。
2.在面对解引用操作符的时候,解开的为首元素地址,也就是第一个整形,大小自然和类型匹配:
int—4。
3.在面对取地址操作符时,无论是否进行加减,都产生为一个字符存储地址,及在x86(32位)环境下为4,x64(64位)环境下为8。
int main()
{
char arr[] = { 'a','b','c','d','e','f' };//[a b c d e f]
printf("%llu\n", sizeof(arr));//6
printf("%llu\n", sizeof(arr + 0));//4/8 arr + 0是数组首元素的地址
printf("%llu\n", sizeof(*arr));//1 - *arr是首元素,首元素是一个字符,大小是一个字节
printf("%llu\n", sizeof(arr[1]));//1 - arr[1]是数组的第二个元素,大小是1个字节
printf("%llu\n", sizeof(&arr));//4/8 &arr是数组的地址
printf("%llu\n", sizeof(&arr + 1));//4/8 &arr + 1是从数组地址开始向后跳过了整个数组产生的一个地址
printf("%llu\n", sizeof(&arr[0] + 1));//4/8 &arr[0] + 1 是数组第二个元素的地址
return 0;
}
1.同上整形数组,在面对数组名为数组大小。
2.在面对解引用操作符的时候,解开的为首元素地址,也就是第一个字符,大小自然和类型匹配:
char—1。
3.在面对取地址操作符时,无论是否进行加减,都产生为一个字符存储地址,及在x86(32位)环境下为4,x64(64位)环境下为8。
int main()
{
char arr[] = "abcdef";
//[a b c d e f \0]
printf("%d\n", sizeof(arr));//7
printf("%d\n", sizeof(arr + 0));//4/8 arr+0是数组首元素的地址
printf("%d\n", sizeof(*arr));//1 - *arr 数组的首元素
printf("%d\n", sizeof(arr[1]));//1 arr[1]数组的第二个元素
printf("%d\n", sizeof(&arr));//4/8 - &arr数组的地址,但是数组的地址依然是地址,是地址大小就是4/8
printf("%d\n", sizeof(&arr + 1));//4/8 - &arr + 1是\0后边的这个地址
printf("%d\n", sizeof(&arr[0] + 1));//4/8 - &arr[0] + 1是数组第二个元素的地址
return 0;
}
1.sizeof在读取字符串中,将字符串的每个字符计算在内,包括结尾默认的'\0'也会计算在内。
2.在面对解引用操作符的时候,解开的为首元素地址,也就是第一个字符,大小自然和类型匹配:
char—1。
3.在面对取地址操作符时,无论是否进行加减,都产生为一个字符存储地址,及在x86(32位)环境下为4,x64(64位)环境下为8。
int main()
{
char* p = "abcdef";
printf("%d\n", sizeof(p));//4/8 p是指针变量,计算的是指针变量的大小
printf("%d\n", sizeof(p + 1));//4/8 p+1是'b'的地址
printf("%d\n", sizeof(*p)); //1 - *p 其实就是'a'
printf("%d\n", sizeof(p[0]));//1 - p[0]-> *(p+0)-> *p
printf("%d\n", sizeof(&p));//4/8 &p 是指针变量p在内存中的地址
printf("%d\n", sizeof(&p + 1));//4/8 - &p+1是跳过p之后的地址
printf("%d\n", sizeof(&p[0] + 1));//4/8 &p[0]是‘a’的地址,&p[0]+1就是b的地址
return 0;
}
1.在指针变量中,sizeof计算的是指针变量的大小。
2.在面对解引用操作符之时,求的是第一个字符的大小。
3.在面对取地址操作符时,无论是否进行加减,都产生为一个字符存储地址,及在x86(32位)环境下为4,x64(64位)环境下为8。