给不明白的大伙说下,下面题目中答案“4或8”的含义。指针也就是地址,在C语言中是同一个说法。地址由计算机地址线产生,32位机器使用32根地址线产生32位电信号(0或1),那么换算字节就是4个字节,64位平台机器同理就是8个字节,所以指针变量的大小取决于是32位机器还是64位机器。现在的机器都是64位,在IDE中既可以改成以32位机器运行,也可以改成以64位机器运行。
int a[] = { 1,2,3,4 };
printf("%zd\n", sizeof(a)); // 16
printf("%zd\n", sizeof(a + 0)); // 4或8
printf("%zd\n", sizeof(*a)); // 4
printf("%zd\n", sizeof(a + 1)); // 4或8
printf("%zd\n", sizeof(a[1])); // 4
printf("%zd\n", sizeof(&a)); // 4或8
printf("%zd\n", sizeof(*&a)); // 16
printf("%zd\n", sizeof(&a + 1)); // 4或8
printf("%zd\n", sizeof(&a[0])); // 4或8
printf("%zd\n", sizeof(&a[0] + 1)); // 4或8
return 0;
逐个分析:
*&a
相当于只有a,代表整个数组。char arr[] = { 'a','b','c','d','e','f' };
printf("%zd\n", sizeof(arr)); // 6
printf("%zd\n", sizeof(arr + 0)); // 4或8
printf("%zd\n", sizeof(*arr)); // 1
printf("%zd\n", sizeof(arr[1])); // 1
printf("%zd\n", sizeof(&arr)); // 4或8
printf("%zd\n", sizeof(&arr + 1)); // 4或8
printf("%zd\n", sizeof(&arr[0] + 1)); // 4或8
逐个分析:
char arr[] = { 'a','b','c','d','e','f' };
printf("%d\n", strlen(arr)); // 不确定的值
printf("%d\n", strlen(arr + 0)); // 不确定的值
printf("%d\n", strlen(*arr)); // 错误
printf("%d\n", strlen(arr[1])); // 错误
printf("%d\n", strlen(&arr)); // 不确定的值
printf("%d\n", strlen(&arr + 1)); // 不确定的值 - 6
printf("%d\n", strlen(&arr[0] + 1)); // 不确定的值 - 1
逐个分析:
char arr[] = "abcdef";
printf("%zd\n", sizeof(arr)); // 7
printf("%zd\n", sizeof(arr + 0)); // 4或8
printf("%zd\n", sizeof(*arr)); // 1
printf("%zd\n", sizeof(arr[1])); // 1
printf("%zd\n", sizeof(&arr)); // 4或8
printf("%zd\n", sizeof(&arr + 1)); // 4或8
printf("%zd\n", sizeof(&arr[0] + 1)); // 4或8
逐个分析:
char arr[] = "abcdef";
printf("%d\n", strlen(arr)); // 6
printf("%d\n", strlen(arr + 0)); // 6
printf("%d\n", strlen(*arr)); // 错误
printf("%d\n", strlen(arr[1])); // 错误
printf("%d\n", strlen(&arr)); // 6
printf("%d\n", strlen(&arr + 1)); // 不确定的值 - 7
printf("%d\n", strlen(&arr[0] + 1)); // 5
逐个分析:
char* p = "abcdef";
printf("%zd\n", sizeof(p)); // 4或8
printf("%zd\n", sizeof(p + 1)); // 4或8
printf("%zd\n", sizeof(*p)); // 1
printf("%zd\n", sizeof(p[0])); // 1
printf("%zd\n", sizeof(&p)); // 4或8
printf("%zd\n", sizeof(&p + 1)); // 4或8
printf("%zd\n", sizeof(&p[0] + 1)); // 4或8
逐个分析:
char* p = "abcdef";
printf("%d\n", strlen(p)); // 6
printf("%d\n", strlen(p + 1)); // 5
printf("%d\n", strlen(*p)); // 错误
printf("%d\n", strlen(p[0])); // 错误
printf("%d\n", strlen(&p)); // 不确定的值
printf("%d\n", strlen(&p + 1)); // 不确定的值 - 4或8
printf("%d\n", strlen(&p[0] + 1)); // 5
逐个分析:
int a[3][4] = { 0 };
printf("%zd\n", sizeof(a)); // 48
printf("%zd\n", sizeof(a[0][0])); // 4
printf("%zd\n", sizeof(a[0])); // 16
printf("%zd\n", sizeof(a[0] + 1)); // 4或8
printf("%zd\n", sizeof(*(a[0] + 1))); // 4
printf("%zd\n", sizeof(a + 1)); // 4或8
printf("%zd\n", sizeof(*(a + 1))); // 16
printf("%zd\n", sizeof(&a[0] + 1)); // 4或8
printf("%zd\n", sizeof(*(&a[0] + 1))); // 16
printf("%zd\n", sizeof(*a)); // 16
printf("%zd\n", sizeof(a[3])); // 16
逐个分析:
其实这些题做下来,会发现核心就是搞懂数组名是什么,写成指针形式是什么样的,那这些题就没啥问题了。
int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int *ptr = (int *)(&a + 1);
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;
int main()
{
printf("%p\n", p + 0x1); // 00100014
printf("%p\n", (unsigned long)p + 0x1); // 00100001
printf("%p\n", (unsigned int*)p + 0x1); // 00100004
return 0;
}
假设p 的值为0x100000。 如下表表达式的值分别为多少?已知,结构体Test类型的变量大小是20个字节(32位机器下:4 + 4 + 2 + 2 + 8)。
解析:首先要知道0x1,虽然是16进制数,但它还是1,说白了就是0x01。
int main()
{
int a[4] = { 1, 2, 3, 4 };
int *ptr1 = (int *)(&a + 1); // 4
int *ptr2 = (int *)((int)a + 1); // 2000000
printf( "%x,%x", ptr1[-1], *ptr2);
return 0;
}
#include
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };
int *p;
p = a[0];
printf( "%d", p[0]); // 1
return 0;
}
这道题其实有坑,要注意看数组初始化,其实里边用的是{},而这里使用了()包含的逗号表达式,那么实际上a数组存储的元素是:1, 3, 5, 0, 0, 0。p指向a的首元素a[0],它是一个一维数组,p[0] == a[0],又数组名等于首元素地址,所以取出的是1。
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;
}
数组指针p指向的是一行为4个整型元素的数组,所以p一次只能操作4个整型。指针-指针得到的结果为两个地址间的元素个数,这道题最难的是%p打印的结果下面逐步分析下过程:
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
int main()
{
char *a[] = {"work","at","alibaba"};
char**pa = a;
pa++;
printf("%s\n", *pa); // at
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); // POINT
printf("%s\n", *--*++cpp+3); // ER
printf("%s\n", *cpp[-2]+3); // ST
printf("%s\n", cpp[-1][-1]+1); // EW
return 0;
}