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

C语言:指针进阶

一、字符指针

	char arr[] = "hello";char* p = "hello";  //p中存的是"hello"字符串首元素的地址printf("%c\n",*arr); //hprintf("%c\n", *p);  //hprintf("%s\n", arr); //"hello",以字符串格式打印,从首元素开始,遇到'\0'结束printf("%s\n", p);  //"hello"*arr = 'x';  //存放在数组中,可以修改//*p = 'y';  //err,char* p = "hello";为常量,不可修改printf("%c\n", *arr); //x//printf("%c\n", *p); 

1.1练习题 

	char str1[] = "hello bit.";char str2[] = "hello bit.";char* str3 = "hello bit.";char* str4 = "hello bit.";if (str1 == str2)printf("str1 and str2 are same\n");elseprintf("str1 and str2 are not same\n");if (str3 == str4)printf("str3 and str4 are same\n");elseprintf("str3 and str4 are not same\n");
//str1 and str2 are not same  
//str3 and str4 are same

str1为自己数组中首元素的地址, str2为自己数组中首元素的地址, str1≠str2;即内存中保存有两份"hello bit.",但存放的地址不同;

str3和str4中对应的字符串均为常量(不可修改),内容一样,又不可修改,在内存中只保留一份,即str3和str4均指向同一个字符串"hello bit."的首字母地址。


二、指针数组

//指针数组
//是数组-数组中存放的是指针(地址)
//int* arr[3];存放整形指针的数组

int a[5] = {1,2,3,4,5};
int b[5] = {2,3,4,5,6};
int c[5] = {3,4,5,6,7};
int* arr[3] = {a,b,c};
for (int i = 0; i < 3; i++)
{for (int j = 0; j < 5; j++)printf("%d ", *(arr[i] + j));//arr[i]+j<=>arr[i][j]printf("\n");
}

三、数组指针

3.1 书写方式

//数组指针 - 是指向数组的指针

	int a = 1;int* pa = &a;char b = 'w';char* pb = &b;int arr[10] = {1,2,3,4,5};//arr,数组名,是首元素arr[0]的地址//&arr,取出的是数组的地址int (*parr)[10] = &arr;//parr数组指针,指向一个数组,存放的是数组的地址//(*parr)确保parr是指针,[10]确保是是数组指针,int:指针指向的数组元素类型为整形double* d[5];double* (*pd)[5] = &d;//(*pd)确保pd是指针,[5]确保是数组指针,double*:指针指向的数组元素类型为浮点型指针

3.2 arr和&arr区别

	int arr[10] = {0};int* p1 = arr;int(*p2)[10] = &arr;printf("%p\n",arr); //0053F9E4printf("%p\n", &arr); //0053F9E4printf("%p\n", p1);  //0053F9E4printf("%p\n", p2);  //0053F9E4printf("%p\n", p1+1); //0053F9E8,比0053F9E4多4printf("%p\n", p2+1); //0053FA0C,比0053F9E4多40

3.3 注意

数组名是数组首元素地址

但有2个例外:

1.sizeof(数组名):计算整个数组的大小,单位是字节

2.&数组名:取出的是整个数组的地址

 3.4 &arr进阶

3.4.1 一维数组

	int arr[5] = {1,2,3,4,5};int(*p)[5] = &arr; //p是指向数组arr的指针for (int i = 0; i < 5; i++)printf("%d ", *((*p) + i)); //*p是数组首元素地址(*p=*(&arr)=arr)//*((*p) + i)<=>*(arr + i)<=>arr[i]

p是指向数组的数组指针,存放的是整个数组的地址;*p解引用表示整个数组首元素的地址;

	int arr1[5] = { 1,2,3,4,5 };int a = 0;int* pa = &a;int* arr2[2] = {pa}; //数组中有2个元素,都是整形指针类型int(*p1)[5] = &arr1;int(*p2)[2] = &arr2;printf("%d\n",**p1);   //1,*p1=arr1,**p1=*arr1=数组首元素printf("%d\n", **(arr2)); //0,*arr2=pa=&a数组首元素,**arr2=*paprintf("%d\n", ***p2); //err(不懂),*p2 = arr2printf("%d\n", *(int*)(**p2)); //0

3.4.2 一维数组传参

void test(int arr[])//ok
{}
void test(int arr[10])//ok
{}
void test(int *arr)//ok
{}
void test2(int *arr[20])//ok
{}
void test2(int **arr)//ok,arr2是数组首元素(地址)的地址,即是地址的地址
{}
int main()
{int arr[10] = {0};int *arr2[20] = {0};test(arr);test2(arr2);
}

3.4.3 二维数组

//arr[3][5]

//1.arr表示二维数组首元素地址,但此首元素为第一行的一维数组,是一个数组指针

//即,arr是一个指向第一行一维数组的数组指针

//2.&arr表示指向整个二维数组的指针

//3.*arr表示指向第一行一维数组首元素的地址《=》arr[0]

//4.**arr表示指向第一行一维数组首元素 <=>*arr[0]<>=arr[0][0]

	int arr[3][5] = {{1,2,3,4,5}, {22,33,44,55,66}, {333,444,555,666,777}};printf("%p\n",*arr);  //第一行一维数组首元素地址,即1的地址printf("%p\n", arr[0]); //第一行一维数组首元素地址,即1的地址printf("%p\n", arr + 1);//第二行一维数组地址printf("%p\n", *(arr + 1)); //第二行一维数组地址首元素地址printf("%p\n", arr[1]); //第二行一维数组地址首元素地址printf("%d\n", **arr); //取出数字1printf("%d\n",*arr[0]);//1printf("%d\n", *(*arr + 1)); //2printf("%d\n", *(arr[0] + 1));  //2, =>arr[0][1]printf("%d\n", *arr[1]);  //22,=>arr[1][0]printf("%d\n", **(arr + 1));//22

 3.4.4二维数组传参

void test(int arr[3][5])//ok
{}
void test(int arr[][])//err
{}
void test(int arr[][5])//ok
{}
void test(int *arr)//err,整形指针
{}
void test(int* arr[5])//err,指针数组
{}
void test(int (*arr)[5])//ok
{}
void test(int **arr)//err,二级指针
{}
int main()
{int arr[3][5] = {0};test(arr); //arr是数组指针
}

四、函数指针

4.1 书写方式

//1.指向函数的指针

//2.存放函数地址的指针

//数组名 !=& 数组名

//函数名 == &函数名(意义相同,写法不同)

int add(int x,int y)
{return x+y;
}
int main()
{printf("%p\n",add); //001613CAprintf("%p\n",&add); //001613CA//pf就是一个函数指针变量int (*pf)(int,int) = &add;//或者 int (*pf)(int,int) = add; pf<==>addint ret = add(3,5); int ret1 = (*pf)(3,5); //*只是一个摆设int ret2 = pf(3,5); printf("%d ",ret1);  //8printf("%d ",ret2);  //8return 0;
}

4.2 练习题

//题一

int main()

{

( * ( void ( * ) ( ) ) 0 ) ( );

//调用0地址处的函数,该函数无参,返回类型是void

//1.void (*) () - 函数指针类型;//void (*p) () - 函数指针变量

//2.( void (*) () ) 0 - 对0进行强制类型转换,被解释为一个函数地址

//3. *( void (*) () ) 0 - 对0地址进行解引用操作

//4.( * ( void ( * ) ( ) ) 0 ) ( ); - 调用0地址处的函数

return 0;

}

//题二

void (* signal(int, void (*) (int) ) ) (int); 

//简化理解:void (*) (int) signal(int, void (*) (int));

//1.signal先和()结合,说明signal是函数名

//2.signal函数的第一个参数类型是int,第二个参数类型是函数指针

//该函数指针,指向一个参数为int,返回类型是void的函数

//3.signal函数的返回类型也是一个函数指针

//该函数指针,指向一个参数为int,返回类型是void的函数


五、函数指针数组 

//函数指针数组

//存放函数指针的数组

int add(int x,int y)
{return x+y;
}
int sub(int x, int y)
{return x - y;
}
int main()
{int (*pf1)(int, int) = add;int (*pf2)(int, int) = sub;int (*pfArr[2])(int, int) = {add,sub}; //pfArr是函数指针数组return 0;
}

int arr[5];

int (*p1)[5] = &arr; //p1是指针,指向整形数组的指针

int* arr[5];

int* (*p2)[5] = &arr; //p2是指针,指向【整型指针数组】的指针

int (*p)(int,int) //p是指针,函数指针

int (*p2[5])(int,int) //p2与[5]先结合,是数组,函数指针数组

int (*(*p3)[5])(int,nit) = &p2; //取出的是函数指针数组的地址

//p3是一个指向【函数指针数组】的指针


六、回调函数 

6.1 定义

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
void menu()
{printf("1.add  2.sub\n");printf("0.exit\n");
}
int add(int a, int b)
{return a + b;
}
int sub(int a, int b)
{return a - b;
}
//回调函数
int calc(int(*pf)(int,int))
{printf("请输入两个数字");int x = 0;int y = 0;scanf_s("%d %d",&x,&y);return pf(x,y);
}
int main()
{int input = 0;do {menu();int ret = 0;printf("请选择:");scanf_s("%d",&input);switch (input){case 1:ret = calc(add);printf("ret=%d\n", ret);break;case 2:ret = calc(sub);printf("ret=%d\n", ret);break;case 0:break;default:printf("输入错误,请重新选择\n");}} while (input);return 0;
}

 七、指针练习

C指针练习题-CSDN博客

C指针sizeof()、strlen()练习题详解-CSDN博客

相关文章:

  • MySQL数据库高阶语句②
  • Golang案例开发之gopacket抓包三次握手四次分手(3)
  • C语言UNIX域套接字CS模型
  • 全局UI方法-弹窗六-自定义弹窗
  • Flask 与小程序 的图片数据交互 过程及探讨研究学习
  • 如何在群晖NAS搭建bitwarden密码管理软件并实现无公网IP远程访问
  • Install Docker
  • 【机器学习】代价函数
  • #Linux(make工具和makefile文件以及makefile语法)
  • spark: 从pulsar中读取数据
  • tcpdump 抓包
  • 基于STELLA系统动态模拟技术及在农业、生态及环境科学中的应用教程
  • WINDOWS设置代理链chain
  • 一文整合工厂模式、模板模式、策略模式
  • 什么是通配符SSL证书?
  • JavaScript-如何实现克隆(clone)函数
  • $translatePartialLoader加载失败及解决方式
  • 〔开发系列〕一次关于小程序开发的深度总结
  • 0x05 Python数据分析,Anaconda八斩刀
  • Java 实战开发之spring、logback配置及chrome开发神器(六)
  • JavaScript HTML DOM
  • JavaScript工作原理(五):深入了解WebSockets,HTTP/2和SSE,以及如何选择
  • KMP算法及优化
  • python_bomb----数据类型总结
  • Vue全家桶实现一个Web App
  • 阿里中间件开源组件:Sentinel 0.2.0正式发布
  • 案例分享〡三拾众筹持续交付开发流程支撑创新业务
  • 浮现式设计
  • 和 || 运算
  • 前端每日实战 2018 年 7 月份项目汇总(共 29 个项目)
  • 浅谈web中前端模板引擎的使用
  • 让你成为前端,后端或全栈开发程序员的进阶指南,一门学到老的技术
  • 深度解析利用ES6进行Promise封装总结
  • 文本多行溢出显示...之最后一行不到行尾的解决
  • 移动端 h5开发相关内容总结(三)
  • 2017年360最后一道编程题
  • # Python csv、xlsx、json、二进制(MP3) 文件读写基本使用
  • (C#)Windows Shell 外壳编程系列9 - QueryInfo 扩展提示
  • (ros//EnvironmentVariables)ros环境变量
  • (超详细)2-YOLOV5改进-添加SimAM注意力机制
  • (二)学习JVM —— 垃圾回收机制
  • (三分钟)速览传统边缘检测算子
  • (转载)Linux网络编程入门
  • *Algs4-1.5.25随机网格的倍率测试-(未读懂题)
  • ./indexer: error while loading shared libraries: libmysqlclient.so.18: cannot open shared object fil
  • .NET Core 通过 Ef Core 操作 Mysql
  • .Net IOC框架入门之一 Unity
  • .net操作Excel出错解决
  • .NET开发不可不知、不可不用的辅助类(一)
  • .NET序列化 serializable,反序列化
  • .py文件应该怎样打开?
  • @manytomany 保存后数据被删除_[Windows] 数据恢复软件RStudio v8.14.179675 便携特别版...
  • @SuppressWarnings注解
  • @test注解_Spring 自定义注解你了解过吗?
  • [.net]官方水晶报表的使用以演示下载