目录
练习:
一维数组:
int a[] = {1,2,3,4};
printf("%d\n",sizeof(a));//16 整个数组大小
printf("%d\n",sizeof(a+0));//4 首元素地址
printf("%d\n",sizeof(*a));//4 首元素的值的大小
printf("%d\n",sizeof(a+1));// 4 a[1]地址的大小
printf("%d\n",sizeof(a[1]));//4 a[1]值的大小
printf("%d\n",sizeof(&a));//4 整个数组地址的大小
printf("%d\n",sizeof(*&a));//16 解引用整个数组的大小
printf("%d\n",sizeof(&a+1));//4
printf("%d\n",sizeof(&a[0]));//4
printf("%d\n",sizeof(&a[0]+1));//4 &a[1]
特别注意:
#include<stdio.h>
int main()
{
int arr[] = { 1,2,3,4 };
int* p = &arr;
printf("%d",sizeof(*&arr));//16
return 0;
}
#include<stdio.h>
int main()
{
int arr[] = { 1,2,3,4 };
int* p = &arr;
printf("%d",sizeof(*p));//4
return 0;
}
上面两个代码看起来很相似,但结果却不一样。这里请注意&arr是有含义的,代表整个数组的地址,当它解引用后是解引用的整个数组的大小,而就算把&arr的值赋给p,p也始终只是一个值而已,故解引用p只能得到首元素的地址。
字符数组:
char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));//6 整个数组
printf("%d\n", sizeof(arr+0));//4 首元素地址
printf("%d\n", sizeof(*arr));//1 首元素
printf("%d\n", sizeof(arr[1]));//1 访问数组1下标
printf("%d\n", sizeof(&arr));// 4 整个数组的地址
printf("%d\n", sizeof(&arr+1));//4 f后面这个地方的地址
printf("%d\n", sizeof(&arr[0]+1));//4 &arr[1]
//strlen(p):从p这个地址开始计算字节长度,直到遇到'\0'停止 其中p的类型是const char*
char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", strlen(arr));//可能没有'\0',随机值X
printf("%d\n", strlen(arr+0));//同上X
printf("%d\n", strlen(*arr));//这是char型与const char*不符,报错
printf("%d\n", strlen(arr[1]));//同上
printf("%d\n", strlen(&arr));//char(*p)[6]=&arr,类型不符,报错
printf("%d\n", strlen(&arr+1));//同上
printf("%d\n", strlen(&arr[0]+1));//从&arr[1]开始,也是随机值,不过是X-1
char arr[] = "abcdef";
printf("%d\n", sizeof(arr));//7 整个数组的长度,还包括'\0'
printf("%d\n", sizeof(arr+0));//4 首元素的地址
printf("%d\n", sizeof(*arr));//1 首元素a的大小
printf("%d\n", sizeof(arr[1]));//1 arr[1]这个元素的大小
printf("%d\n", sizeof(&arr));//4 整个数组的地址的大小
printf("%d\n", sizeof(&arr+1))//4 f后面地址的大小
printf("%d\n", sizeof(&arr[0]+1));//4 &arr[1]的大小
char arr[] = "abcdef";
printf("%d\n", strlen(arr));//6 整个数组的大小
printf("%d\n", strlen(arr+0));//6 给首元素的地址,整个数组的大小
printf("%d\n", strlen(*arr));//char型,类型不符,报错
printf("%d\n", strlen(arr[1]));//同上
printf("%d\n", strlen(&arr));//char(*p)[6]=&arr类型不符,报错
printf("%d\n", strlen(&arr+1));//同上
printf("%d\n", strlen(&arr[0]+1));//5 从arr[1]开始到'\0'
char *p = "abcdef";
printf("%d\n", sizeof(p));//4 首元素的地址
printf("%d\n", sizeof(p+1));//4 b元素的地址
printf("%d\n", sizeof(*p));//1 访问a元素的大小
printf("%d\n", sizeof(p[0]));1 同上
printf("%d\n", sizeof(&p));4 整个字符串的地址的大小
printf("%d\n", sizeof(&p+1));//4 f后面的地址的大小
printf("%d\n", sizeof(&p[0]+1));//4 &p[1]的大小
char *p = "abcdef";
printf("%d\n", strlen(p));//6 整个字符串的大小
printf("%d\n", strlen(p+1));//5 从b开始到\0的长度
printf("%d\n", strlen(*p));//类型不符,char 型
printf("%d\n", strlen(p[0]));//同上
printf("%d\n", strlen(&p));//整个字符串的地址,类型不符
printf("%d\n", strlen(&p+1));//同上
printf("%d\n", strlen(&p[0]+1));//5
int main()
{
char *str1 = "abcdef";
char *str2 = "abcdef\0abcdef";
char str3[] = "abcdef";
//char str33[] = {'a','b','c'};
char str4[] = "abcdef\0abcdef";
printf("%d\n", strlen(str1));//6
printf("%d\n", strlen(str2));//6
printf("%d\n", strlen(str3));//6
printf("%d\n", strlen(str4));//6
printf("%d\n", sizeof(str1));// 4
printf("%d\n", sizeof(str2));// 4
printf("%d\n", sizeof(str3));// 7
printf("%d\n", sizeof(str4));//14
return 0;
}
int a[3][4] = {0};//a是一维数组的地址
printf("%d\n",sizeof(a));//48 12*4整个数组大小
printf("%d\n",sizeof(a[0][0]));//4
printf("%d\n",sizeof(a[0]));//16 sizeof(数组名)代表整个一维数组
printf("%d\n",sizeof(a[0]+1));//4 数组名代表数组首元素的地址,数组名+1表示&a[0][1]
printf("%d\n",sizeof(*(a[0]+1)));//4 等价于*(*(a+0)+1)=a[0][1]
printf("%d\n",sizeof(a+1));//4 一行的地址
printf("%d\n",sizeof(*(a+1)));//16 对一行整个数组的地址解引用,是整个数组的大小=a[1]
printf("%d\n",sizeof(&a[0]+1));//4 一行的地址 &a[1]
printf("%d\n",sizeof(*(&a[0]+1)));//16 一行的地址解引用,引用整个数组大小
printf("%d\n",sizeof(*a));//16 0行数组的地址,解引用引用整个0行数组
printf("%d\n",sizeof(a[3]));//16
总结: 数组名的意义:
1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
3. 除此之外所有的数组名都表示首元素的地址。
int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int *ptr = (int *)(&a + 1);//数组指针强转为int *类型
printf( "%d,%d", *(a + 1), *(ptr - 1));//2 5
return 0;
}
struct Test
{
int Num;
char *pcName;
short sDate;
char cha[2];
short sBa[4];
}*p;//20
int main()
{
p = 0x00100000;
printf("%p\n", p + 0x1);//0x00100014// 16+4
printf("%p\n", (unsigned long)p + 0x1);//0x00100001
printf("%p\n", (unsigned int*)p + 0x1);//0x00100004
return 0;
}
注意:变量加一,要看该变量类型,若该变量为指针,则+相应指针的大小,若该变量为整型,则就加一。
int main()
{
int a[4] = { 1, 2, 3, 4 };
int *ptr1 = (int *)(&a + 1);
int *ptr2 = (int *)((int)a + 1);
//*(ptr1-1) 4
printf( "%x,%x", ptr1[-1], *ptr2);//4 02 00 00 00
return 0;
}
#include <stdio.h>
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int *p;
p = a[0];
printf( "%d", p[0]);// 1
return 0;
}
int main()
{
int a[5][5];
int(*p)[4];
p = a;
printf( "%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);//fffffffc -4
return 0;
}
int main()
{
int aa[2][5] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 };
int *ptr1 = (int *)(&aa + 1);
int *ptr2 = (int *)(*(aa + 1));
printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));// 10 5
return 0;
}
#include <stdio.h>
int main()
{
char *a[] = {"work","at","alibaba"};
char**pa = a;
pa++;
printf("%s\n", *pa);
return 0;
}
int main()
{
char *c[] = {"ENTER","NEW","POINT","FIRST"};
char**cp[] = {c+3,c+2,c+1,c};
char***cpp = cp;
printf("%s\n", **++cpp);
printf("%s\n", *--*++cpp+3);
printf("%s\n", *cpp[-2]+3);
printf("%s\n", cpp[-1][-1]+1);
return 0;
}