10.3 指针进阶_代码分析
代码分析
- 9. 指针和数组代码解析
- 一维数组
- 字符数组
- 字符串
- 二维数组
- 10. 指针代码分析
- eg1
- eg2
- eg3
- eg4
- eg5
- eg6
- eg7
- eg8
10.1 指针进阶_数组指针
10.2 指针进阶_函数指针
9. 指针和数组代码解析
数组名arr是首元素地址
例外:
1. sizeof(arr),计算整个数组的大小,sizeof内部单独放一个数组名,数组名表示整个数组。
2. &arr,取出的是 数组的地址。&数组名,数组名表示 整个数组。
除此1,2两种情况之外,所有的数组名都表示数组首元素的地址
一维数组
int a[] = { 1,2,3,4 };
printf("%d\n", sizeof(a));//数组大小-->4*4=16 byte
printf("%d\n", sizeof(a+0));//a首元素地址,a+0-->4/8 byte
printf("%d\n", sizeof(*a));//a首元素地址,解引用*a,首元素大小-->4 byte
printf("%d\n", sizeof(a+1));//a+1-->4/8 byte
printf("%d\n", sizeof(a[1]));//第二个元素大小-->4 byte
printf("%d\n", sizeof(&a));//数组地址&a-->4/8 byte
printf("%d\n", sizeof(*&a));
//*&a --> a --> 数组地址&a,解引用*&a,数组大小大小-->16 byte
//&a --> int(*)[4] -->类型
// --> *&a --> a[4] 16 byte
printf("%d\n", sizeof(&a+1));//相对于&a,跳过了整个数组,地址大小-->4/8 byte
printf("%d\n", sizeof(&a[0]));//首元素地址-->4/8 byte
printf("%d\n", sizeof(&a[0]+1));//第二个元素地址-->4/8 byte
字符数组
char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));//整个数组大小 --> 6 byte
printf("%d\n", sizeof(arr+0));//首元素地址 --> 4/8 byte
printf("%d\n", sizeof(*arr));//首元素大小 --> 1 byte
printf("%d\n", sizeof(arr[1]));//第二个元素大小 --> 1 byte
printf("%d\n", sizeof(&arr));//数组地址 --> 4/8byte
printf("%d\n", sizeof(&arr+1));//相对于&arr,跳过了整个数组,地址-->4/8 byte
printf("%d\n", sizeof(&arr[0]+1));//首元素地址+1,第二个元素地址-->4/8 byte
printf("%d\n", strlen(arr));//strlen读到\0才结束,统计\0前字符串个数,随机值
printf("%d\n", strlen(arr+0));//首元素地址arr+0,随机值
printf("%d\n", strlen(*arr));//err,strlen参数部分需要传地址
//*arr是首元素a,a的ASCII码位97,strlen从97开始统计字符串长度,非法访问内存
printf("%d\n", strlen(arr[1]));//err
printf("%d\n", strlen(&arr));//&arr --> char(*)[6],数组地址,随机值
printf("%d\n", strlen(&arr+1));//随机值
printf("%d\n", strlen(&arr[0]+1));//第二个元素地址,随机值
char arr[] = "abcdef";//[a b c d e f \0]
printf("%d\n", sizeof(arr));//数组大小 --> 7
printf("%d\n", sizeof(arr+0));//首元素地址 --> 4/8
printf("%d\n", sizeof(*arr));//首元素 --> 1
//*arr --> *(arr + 0) --> arr[0]
printf("%d\n", sizeof(arr[1]));//第二元素 --> 1
printf("%d\n", sizeof(&arr));//数组地址 --> 4/8
printf("%d\n", sizeof(&arr+1));//数组地址+1 --> 4/8
printf("%d\n", sizeof(&arr[0]+1));//第二元素地址 --> 4/8
printf("%d\n", strlen(arr));//数组地址 6
printf("%d\n", strlen(arr+0));//首元素地址 6
printf("%d\n", strlen(*arr));//err 首元素a
printf("%d\n", strlen(arr[1]));//err 首元素a
printf("%d\n", strlen(&arr));//数组地址 6
printf("%d\n", strlen(&arr+1));//数组地址+1 random
printf("%d\n", strlen(&arr[0]+1));//第二元素地址 5
字符串
char* p = "abcdef";//[a b c d e f \0],p存放a的地址
printf("%d\n", sizeof(p));//指针p 4/8
printf("%d\n", sizeof(p+1));//b的地址 4/8
printf("%d\n", sizeof(*p));//a 1
printf("%d\n", sizeof(p[0]));//a 1
//*p --> *(p + 0) --> p[0]
printf("%d\n", sizeof(&p));//p的地址 4/8
//&p --> char**
printf("%d\n", sizeof(&p+1));//p的地址+1 4/8
printf("%d\n", sizeof(&p[0]+1));//b的地址+1 4/8
printf("%d\n", strlen(p));//a的地址 6
printf("%d\n", strlen(p+1));//b的地址 5
printf("%d\n", strlen(*p));//err,a
printf("%d\n", strlen(p[0]));//err,a
printf("%d\n", strlen(&p));//p的地址 random
printf("%d\n", strlen(&p+1));//p的地址+1 random
printf("%d\n", strlen(&p[0]+1));//b的地址 5
二维数组
int a[3][4] = {0};
printf("%d\n",sizeof(a));//数组a大小 3*4*4=48
printf("%d\n",sizeof(a[0][0]));//[0][0]元素 4
printf("%d\n",sizeof(a[0]));//[0]行数组大小 4*4=16
//每行都是二维数组的一个元素,一维数组的数组
//a[0]是第一行一维数组的 数组名
printf("%d\n",sizeof(a[0]+1));//第1行数组第二个元素地址 4/8
printf("%d\n",sizeof(*(a[0]+1)));//第2行数组首元素大小 4
printf("%d\n",sizeof(a+1));//a是首元素地址,是第一行的地址,a+1是第2行地址 4/8
printf("%d\n",sizeof(*(a+1)));//第2行数组大小 4*4=16
printf("%d\n",sizeof(&a[0]+1));//&a是第1行的地址,a+1是第2行地址 4/8
printf("%d\n",sizeof(*(&a[0]+1)));//第2行数组大小 4*4=16
printf("%d\n",sizeof(*a));//第1行数组大小 4*4=16
//a 首元素地址,第1行地址
//*a 第1行
//*a --> *(a+0) --> a[0]
printf("%d\n",sizeof(a[3]));//第4行数组大小 4*4=16
//a[3] --> int [4]
//sizeof()只关注类型,不会计算
10. 指针代码分析
eg1
int main()
{
int a[5] = { 1, 2, 3, 4, 5 };
int *ptr = (int *)(&a + 1);//&a 取出 数组地址,&a+1 向后跳一个数组长度 的地址
printf( "%d,%d", *(a + 1), *(ptr - 1));
//(a+1) 首元素地址+1,第二元素地址。*(a+1) --> 2
//ptr 是a[5]向后跳一个数组长度 的地址
//(ptr - 1) 向前访问4byte --> 5
return 0;
}
eg2
//结构体的大小是20个字节
struct Test
{
int Num;//4
char *pcName;//1
short sDate;//2
char cha[2];//1
short sBa[4];//2
}*p = (struct Test*)0x100000;
//结构体Test类型的变量大小是20个字节
int main()
{
printf("%p\n", p + 0x1);//跳过一个结构体 20byte
//0x100000 + 0x000014 = 0x100014
printf("%p\n", (unsigned long)p + 0x1);//整形+1byte --> 0x100001
printf("%p\n", (unsigned int*)p + 0x1);//跳过+4byte --> 0x100004
return 0;
}
eg3
int main()
{
int a[4] = { 1, 2, 3, 4 };
//小端 [01 00 00 00| 02 00 00 00 00| 03 00 00 00| 04 00 00 00]
int *ptr1 = (int *)(&a + 1);//&a 取出数组地址,&a+1 向后跳一个数组长度 的地址
int *ptr2 = (int *)((int)a + 1);//(int)a a地址的值+ 1byte
printf( "%x,%x", ptr1[-1], *ptr2);//%x --> 16进制打印
//ptr1[-1] --> 4
//*ptr2 --> 01 |00 00 00 02| -->2000000
return 0;
}
eg4
#include <stdio.h>
int main()
{
int a[3][2] = { (0, 1), (2, 3), (4, 5) };//(0,1) --> 1. {0,1} is right
//{{1,3}, {5,0}, {0,0}};
int *p;
p = a[0];//a是首元素地址,即第一行地址,a[0][0]的地址
printf( "%d", p[0]);//p[0] --> *(p+0) -->*p --> 1
return 0;
}
eg5
int main()
{
int a[5][5];
int(*p)[4];
p = a;
printf( "%p,%d\n",
&p[4][2] - &a[4][2], // -4
&p[4][2] - &a[4][2]);
return 0;
}
-4 -->
10000000 00000000 00000000 00000100
11111111 11111111 11111111 11111011
1111 1111 1111 1111 1111 1111 1111 1100
F F F F F F F C
0xFFFFFFFC
eg6
int main()
{
int a[2][5] = { 1, 2, 3, 4, 5,
6, 7, 8, 9, 10 };
int *ptr1 = (int *)(&a + 1);//数组地址+1,跳过一个数组的大小
int *ptr2 = (int *)(*(a + 1));//第一行地址+1=第二行地址,*解引用 --> a[1][0]
//a + 1 类型为 int (*)[5] 指向二维数组 a 的第二行
//*(a + 1) 表示指向二维数组 a 第二行首元素的指针,它指向的是 a[1][0]
//*(a + 1) --> a[1]数组名 --> 首元素地址&a[1][0]
printf( "%d,%d", *(ptr1 - 1), *(ptr2 - 1));
//ptr1 -1 --> 10
//ptr2 -1 --> 5
return 0;
}
eg7
#include <stdio.h>
int main()
{
char* a[] = {"work","at","sangfor"};
//char* p = "sangfor";//存放字符串首字符地址
char** pa = a;//pa存放数组名
pa++;// wo --> a
printf("%s\n", *pa);
return 0;
}
eg8
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);
printf("%s\n", *cpp[-2]+3);
printf("%s\n", cpp[-1][-1]+1);
return 0;
}
printf("%s\n", ** ++cpp);//POINT 注意:cpp值已经更改
printf("%s\n", *--*++cpp+3);//ER
printf("%s\n", *cpp[-2]+3);//ST
printf("%s\n", cpp[-1][-1]+1);//EW
10.1 指针进阶_数组指针
10.2 指针进阶_函数指针