函数指针数组指针数组传参的本质字符指针

🚀 作者:阿辉不一般
🚀 你说呢:不服输的你,他们拿什么赢
🚀 专栏:爱上C语言
🚀作图工具:draw.io(免费开源的作图网站)
请添加图片描述

如果觉得文章对你有帮助的话,还请点赞,关注,收藏支持博主,如有不足还请指点,博主及时改正,感谢大家支持!!!

文章目录

  • 🚀前言
  • 🚀字符指针变量
  • 🚀数组指针变量
  • 🚀数组传参的本质
    • ✈️一维数组传参的本质
    • ✈️二维数组传参的本质
    • 总结
  • 🚀函数指针变量
  • 🚀指针变量是什么?

🚀前言

在阿辉上一篇博客指针的基础篇中我们了解到指针的一些基础知识

  • 指针变量是用来存放地址的变量,通过指针可以找到所存地址指向的空间
  • 指针变量的大小与平台有关,64位/32位平台大小为8字节/4个字节
  • 指针变量的类型决定了指针变量所指向的内存空间的类型和大小以及指针加减整数时移动的字节数
  • 指针的运算
  • 多级指针

有了以上对于指针的基础了解,那么今天阿辉将为大家介绍C语言的指针部分,包括字符指针、数组指针、数组传参的本质以及函数指针,关注阿辉不迷路哦 😘 ,内容干货满满😋,接下来就跟着阿辉一起学习吧👊

🚀字符指针变量

在指针的类型中我们知道有⼀种指针类型为字符指针 char*
一般我们这么用:

int main()
{char ch = 'w';char *pc = &ch;*pc = 'w';return 0;
}

其实它还可以这么用:

int main()
{char* pstr = "hi bro";printf("%s\n", pstr);return 0;
}

可能大家认为ptr里面存的是hi bro,实际上ptr里面存的是字符串的首元素地址也就是h的地址,这里大家有没有发现其实这和字符数组是一样的,比如char arr[] = “hi bro”数组名arr 也是首元素h的地址,我们不妨把字符指针理解成字符数组,但真的这么简单吗?他们还是有两点不同,我们接着看👇
第一点不同:
在这里插入图片描述
我们可以看到当我们去改hi bro的内容时编译器直接报错,这是因为hi bro是常量字符串,而常量字符串被存储在程序的只读数据段(.rodata)中,这个数据段是只读的,意味着其中的数据在程序执行期间是不可修改的,而字符数组是可以修改的
第二点不同:

int main()
{char* str1 = "hi bro";char* str2 = "hi bro";char arr1[] = "hi bro";char arr2[] = "hi bro";if (str1 == str2)printf("str1与str2空间相同\n");elseprintf("str1与str2空间不相同\n");if(arr1 == arr2)printf("arr1与arr2空间相同\n");elseprintf("arr1与arr2空间不相同\n");return 0;
}

输出:

str1与str2空间相同
arr1与arr2空间不相同

其实很简单,这里str1str2指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域,当指针指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。所以arr1arr2不同,str1str2相同

🚀数组指针变量

在之前阿辉的数组篇中讲到关于数组名的理解,数组名是数组首元素的地址,而取地址数组名是整个数组的地址,那取地址数组名既然是地址那它应该存在什么类型的指针中呢?没错,就是数组指针
例如:

	int arr[5] = { 0,1,2,3,4 };int(*p)[5] = &arr;p是变量名int(*)[5]是p变量的类型*指的是p是指针int[5]表示p指向的空间的类型是5int类型变量的数组[]的优先级高于*,所以要把*p用括号()括起来表明p是指针我们对p解引用,*p就是arr也就是首元素地址我们对*p再次解引用,就访问到数组第一个元素**p等价于arr[0]通过*(*p + i)我们就可以遍历数组arr[5]

那数组指针有何用?别急我们接着看👊

🚀数组传参的本质

✈️一维数组传参的本质

我们知道一维数组的数组名是首元素地址,当我们传一个数组给函数时,我们不会去在开辟一块同样的空间,而是通过首元素地址访问数组,这样有效的避免了空间的浪费
函数接收一维数组有三种方式,例如:

void test(int a[]);
void test(int a[5]);
void test(int* a);

写法确实有三种,不过前两种人模狗样的是数组(毕竟传数组用数组接受很容易理解)本质上还是指针,包括[5]这个5屁用没有
注意
我个人的一点总结:当一维数组名作为首元素地址传给函数后,函数内部接收地址的形参不再具有数组名的性质,仅仅是一个普通的指针
怎么理解呢,我们接着看👇

void test(int a[])
{&a;sizeof(a);
}
int main()
{int a[5] = { 1,2,3,4,5 };test(a);return 0;
}

在这里插入图片描述
上图是x86环境下的调试结果,我们看到&a类型是int * *是二级指针而并非数组指针,sizeof(a)的值是4,如果a还具有数组名的特性,我们知道&a将是int(*)[5]类型的数组指针,而sizeof(a)的值将是20。这恰恰说明了当一维数组名作为首元素地址传给函数后,函数内部接收地址的形参不再具有数组名的性质,仅仅是一个普通的指针

✈️二维数组传参的本质

void test(int arr[][4]);
void test(int arr[3][4]);
行可以省略,列不能省略

那函数接收二维数组还有没有其他的方式?
二维数组的数组名怎么理解呢?二维数组可以理解为数组的数组,二维数组的每一行理解为二维数组的一个元素,二维数组的数组名同样是首元素地址,只不过二维数组的首元素是第一行,比如下图中二维数组arr[3][4]首元素是绿色的那一行一维数组,所以二维数组的数组名arr表示一个int(*)[4]类型的数组指针

请添加图片描述
那么二维数组的传参就可以这么写void test(int (*arr)[4])
例子:

void test(int (*arr)[4])
{int i = 0;int j = 0;for (i = 0; i < 3; i++){for (j = 0; j < 4; j++){printf("%d ", *(*(arr + i) + j));}printf("\n");}
}
int main()
{int arr[3][4] = { {1,2,3,4},{2,3,4,5},{3,4,5,6} };test(arr);return 0;
}

在这里插入图片描述

总结

  • 一维数组的数组名是存放该数组首元素的一个指针,二维数组的数组名是存放该数组首元素的一个数组指针
  • 一个数组int arr[5]={1,2,3,4,5},你可以用arr[1]*(arr + 1)两种方式访问数组的了第二个元素,甚至可以1[arr]访问了解一下不建议使用。 *(arr+i)等价于arr[i]

🚀函数指针变量

函数指针变量顾名思义是存放函数地址的指针,函数也有地址吗?
没错函数也有地址,函数的函数名就是函数的地址
我们来看一段代码👇

#include <stdio.h>
void test()
{printf("hehe\n");
}
int main()
{printf("test: %p\n", test);printf("&test: %p\n", &test);return 0;
}

输出:

test:0xff44f68a
&test:0xff44f68a

我们看到test&test打印出来的地址是一样的,不仅打印出来的是一样的,它俩本质也是一样的,这俩等价,这个&多少有点多余 😆
那函数指针如何创建呢?其实与数组指针类似

int add(int x, int y)
{return x + y;
}int main()
{int(*p)(int, int) = add;p是函数指针变量名*表示p是一个指针int(*)(int,int)是p的类型int(int,int)表示p所指向的空间是函数,函数的返回类型是int而且有两个int类型的形参return 0;
}

那函数指针变量有什么用呢?
我们可以通过函数指针变量来调用函数,用上面的函数演示

int ret = (*p)(3,5);
*P要用括号括起来,因为函数调用操作符的优先级更高

其实p*padd以及&add这四个等价
所以调用add这个函数(*p)(3,5)p(3,5)这俩都可以,取地址和解引用都挺多余😆

🚀指针变量是什么?

大家看到这个标题或许很懵,指针变量不就是用来存放地址的变量嘛,讲这么多了还问?

int add(int x, int y)
{return x + y;
}int main()
{int(*padd)(int, int) = add;int a = 0;int* pa = &a;int arr[5] = { 1,2,3,4,5 };int(*parr)[5] = &arr;return 0;
}

上面这段代码中paddint( * )(in,int)类型的函数指针变量,paint类型的指针变量,parrint( * )[5]类型的数组指针变量,其实add也是 int( * )(in,int)类型的函数指针变量,&a也是int类型的指针变量,&arr也是int( * )[5]类型的数组指针变量

地址是类似于0xff40688a这样的玩意,add、&a、&arr里面存的不也是地址嘛,SO他们也是指针变量


到这里,阿辉今天对于C语言中一些特殊类型的指针的分享就结束了,希望这篇博客能让大家有所收获, 如果觉得阿辉写得不错的话,记得给个赞呗,你们的支持是我创作的最大动力🌹
请添加图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/174259.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

XSTRING与STRING之间的互转,base64,长文本,科学计数法

XSTRING的介绍 SAP ABAP 理解RAWSTRING(XSTRING) 类型-腾讯云开发者社区-腾讯云 XString,String以及SString 类型区别 | 摆渡SAP SAP ABAP 理解RAWSTRING(XSTRING) 类型 RAWSTRING 和 STRING 类型具有可变长度。可以指定这些类型的最大长度&#xff0c;但没有上限。 SSTRI…

代码浅析DLIO(二)---预积分与单点去畸变

0. 简介 我们刚刚了解过DLIO的整个流程&#xff0c;我们发现相比于Point-LIO而言&#xff0c;这个方法更适合我们去学习理解&#xff0c;同时官方给出的结果来看DLIO的结果明显好于现在的主流方法&#xff0c;当然指的一提的是&#xff0c;这个DLIO是必须需要六轴IMU的&#x…

【ZYNQ】SD 卡读写及文件扫描实验

SD 卡控制器&#xff08;SD/SDIO Controller&#xff09; ZYNQ 中的 SD 卡控制器符合 SD2.0 协议规范&#xff0c;接口兼容 eMMC、MMC3.31、SDIO2.0、SD2.0、SPI&#xff0c;支持 SDHC、SDHS 器件。SD 卡控制器支持 SDMA&#xff08;单操作 DMA&#xff09;、ADMA1&#xff08…

数据结构-顺序表

文章目录 线性表概念顺序表静态顺序表动态顺序表 总结 线性表概念 线性表是最基本、最简单、也是最常用的一种数据结构&#xff0c;常见的线性表:顺序表、链表、栈、队列、字符串…线性表&#xff08;linear> list&#xff09;是数据结构的一种&#xff0c;一个线性表是n个具…

蚂蚁庄园小课堂答题今日答案最新

蚂蚁庄园小课堂答题今日答案最新 温馨提醒&#xff1a;由于本文章会停留在一个固定的更新时间上&#xff0c;包含当前日期最新的支付宝蚂蚁庄园小课堂答题今日答案。如果您看到这篇文章已成为过去时&#xff0c;请按下面的方法进入查看天天保持更新的最新今日答案&#xff1b; …

Linux 网络通信

(一)套接字Socket概念 Socket 中文意思是“插座”&#xff0c;在 Linux 环境下&#xff0c;用于表示进程 x 间网络通信的特殊文件 类型。本质为内核借助缓冲区形成的伪文件。 既然是文件&#xff0c;那么理所当然的&#xff0c;我们可以使用文件描述符引用套接字。Linux 系统…

Windows11安装后跳过联网登录

Windows11安装后跳过联网登录 实验设备&#xff1a; VMware17Pro虚拟机中使用Windows11镜像安装Windows11操作系统&#xff0c;并且在虚拟机中测试跳过联网登录。 步骤 说明&#xff1a;物理卸载网卡&#xff08;在虚拟机上禁用网卡&#xff09;没用 思路&#xff1a; sh…

8.统一异常处理 + 统一记录日志

目录 1.统一异常处理 2.统一记录日志 1.统一异常处理 在 HomeController 类中添加请求方法&#xff08;服务器发生异常之后需要统一处理异常&#xff0c;记录日志&#xff0c;然后转到 500 页面&#xff0c;需要人工处理重定向到 500 页面&#xff0c;提前把 500 页面请求访问…

经典神经网络——AlexNet模型论文详解及代码复现

一、背景 AlexNet是在2012年由Alex Krizhevsky等人提出的&#xff0c;该网络在2012年的ImageNet大赛上夺得了冠军&#xff0c;并且错误率比第二名高了很多。Alexnet共有8层结构&#xff0c;前5层为卷积层&#xff0c;后三层为全连接层。 论文地址&#xff1a;ImageNet Classif…

ModuleNotFoundError: No module named ‘mdtex2html‘ module已经安装还是报错,怎么办?

用streamlit运行ChatGLM/basic_model/web_demo.py的时候&#xff0c;出现了module not found&#xff1a; ModuleNotFoundError: No module named mdtex2html Traceback: File "/home/haiyue/.local/lib/python3.10/site-packages/streamlit/runtime/scriptrunner/script…

【阿里云】图像识别 智能分类识别 增加网络控制功能点(三)

一、增加网络控制功能 实现需求TCP 心跳机制解决Soket异常断开问题 二、Linux内核提供了通过sysctl命令查看和配置TCP KeepAlive参数的方法。 查看当前系统的TCP KeepAlive参数修改TCP KeepAlive参数 三、C语言实现TCP KeepAlive功能 四、setsockopt用于设置套接字选项的系…

Qt4利用MVC开发曲线数据编辑器

目录 1 需求 2 开发流程 1 搭建框架 2 构造函数 3 打开工程 4 实现应用程序参数加载 5 QCustomPlot和TableView的联动 6 数据的可视化修改 7 列表点击事件事先键盘控制 8 表格实现复制&#xff0c;粘贴&#xff0c;删除等一系列功能 9 曲线实现自适应范围和统一范围…

【JMeter】运行方式

第一种&#xff1a; 使用GUI 操作&#xff1a; 在JMeter界面菜单导航上点击运行按钮 一般用作创建TestPlan和调试脚本增加java堆空间来满足测试环境 第二种&#xff1a;使用CLI(Command Line) 性能测试一般请求量比较大&#xff0c;为了节省资源 CLI参数用法&#xff1a; 字段…

Flask Echarts 实现历史图形查询

Flask前后端数据动态交互涉及用户界面与服务器之间的灵活数据传递。用户界面使用ECharts图形库实时渲染数据。它提供了丰富多彩、交互性强的图表和地图&#xff0c;能够在网页上直观、生动地展示数据。ECharts支持各种常见的图表类型&#xff0c;包括折线图、柱状图、饼图、散点…

[Spring] 字节一面~Spring 如何解决循环依赖问题 以及 @resource 与 @autowire 同时存在时谁生效

文章目录 Spring 如何解决循环依赖问题resource 与 autowire 同时存在时谁生效 Spring 如何解决循环依赖问题 Spring在实例化一个bean的时候&#xff0c;是首先递归实例化其所依赖的所有bean&#xff0c;直到某个bean没有依赖其他bean&#xff0c;此时就会将该实例返回&#x…

【JavaWeb】Servlet

Servlet 文章目录 Servlet一、简介二、开发流程三、生命周期四、ServletConfig和ServletContext五、HttpServletRequest常见API六、HttpServletResponse常见API七、请求转发和响应重定向7.1 概述7.2 请求转发7.3 响应重定向 八、请求与响应乱码问题8.1 GET与POST请求乱码8.2 响…

内网穿透的应用-Jupyter Notbook+cpolar内网穿透实现公共互联网访问使用数据分析工作

文章目录 1.前言2.Jupyter Notebook的安装2.1 Jupyter Notebook下载安装2.2 Jupyter Notebook的配置2.3 Cpolar下载安装 3.Cpolar端口设置3.1 Cpolar云端设置3.2.Cpolar本地设置 4.公网访问测试5.结语 1.前言 在数据分析工作中&#xff0c;使用最多的无疑就是各种函数、图表、…

五、Lua流程控制与函数

一、流程控制 &#xff08;一&#xff09;含义 Lua 编程语言流程控制语句通过程序设定一个或多个条件语句来设定。在条件为 true 时执行指定程序代码&#xff0c;在条件为 false 时执行其他指定代码。 &#xff08;二&#xff09;原型 if (成立) then执行体1else执行体2 end…

字符串入门算法题!

概述 字符串和数组一样算是比较简单的题目&#xff0c;正适合打算法基础&#xff0c;一定要认真对待&#xff01;&#xff01;&#xff01; 字符串类型的算法问题可以分为简单、中等和困难的难度级别&#xff0c;基础类型一些基本的字符串处理问题&#xff0c;如字符串的拼接…

自动化部署 扩容openGauss —— Ansible for openGauss

前言 大家好&#xff0c;今天我们为大家推荐一套基于Ansible开发的&#xff0c;自动化部署及扩容openGauss的脚本工具&#xff1a;Ansible for openGauss&#xff08;以下简称 AFO&#xff09;。 通过AFO&#xff0c;我们只需简单修改一些配置文件&#xff0c;即可快速部署多种…