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

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

相关文章:

  • Java 中如何创建多线程?
  • 2025年能源工作指导意见重点内容
  • fps项目总结:关于攻击与受击
  • 【医学影像 AI】使用血管特征量化自动诊断早产儿视网膜病变中的附加病变
  • C语言:结构体的内存对齐方式
  • vite+react+ts如何集成redux状态管理工具,实现持久化缓存
  • MATLAB中asManyOfPattern函数用法
  • 代理对象中使用this
  • anolis8.9-k8s1.32-系统基本配置
  • Linux--基本指令2
  • 使用Python简单自动地生成图文并茂的网页文件(WEB数据可视化)
  • C# 类库打包dll文件
  • 操作系统之文件系统
  • 一次有趣的前后端跨越排查
  • MobileViTv3模型详解及代码复现
  • vscode接入ai插件(免费版)
  • 2025.3.1学习内容----网络编程
  • 蓝桥杯 门牌制作
  • 儿童英语启蒙规划
  • 分布式拒绝服务(DDoS)攻击检测系统的设计与实现
  • 电商网站建设需求分析书/附子seo教程
  • 外贸seo建站/百度推广代理公司哪家好
  • 做网站需要注意的问题/软件开发网站
  • 东莞最新通报最新/seo关键词优化方法
  • 关于酒店网站规划建设方案书/seo的搜索排名影响因素有哪些
  • 东莞专业做淘宝网站/关键词推广效果