当前位置: 首页 > article >正文

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 指针进阶_函数指针


http://www.kler.cn/a/568480.html

相关文章:

  • 自学微信小程序的第七天
  • hive之LEAD 函数详解
  • 【重构小程序】升级JDK1.8、SpringBoot2.x 到JDK17、Springboot 3.x(一)
  • 算法刷题-2025年02月26日
  • JMeter 实战项目脚本录制最佳实践(含 BadBoy 录制方式)
  • 【Maven】基于IDEA进行Maven工程的创建、构建
  • 【前端场景面试】登录鉴权实现方式详解
  • 《论软件测试中缺陷管理及其应用》审题技巧 - 系统架构设计师
  • 【C++/数据结构】队列
  • uni-app 全局请求封装:支持 Promise,自动刷新 Token,解决 401 过期问题
  • 【数据结构】堆与二叉树
  • 基于SpringBoot和Leaflet的行政区划地图掩膜效果实战
  • 人工智能领域顶级期刊
  • 【经验帖】退出MobaXterm后桌面生成两个配置文件
  • Apifox 后置操作中发送请求
  • 【ISP】AF自动对焦
  • C++错误Call to implicitly-deleted default constructor of ‘SerialPortConfig‘
  • 图数据库Neo4j面试内容整理-水平扩展
  • 基因型—环境两向表数据分析——基因型评价
  • JS逆向-233网校sid参数和sign参数