【C语言】指针的进阶篇,深入理解指针和数组,函数之间的关系

欢迎来CILMY23的博客喔,本期系列为【C语言】指针的进阶篇,深入理解指针和数组,函数之间的关系,图文讲解其他指针类型以及指针和数组,函数之间的关系,带大家更深刻理解指针,以及数组+指针,指针和函数的用法,感谢观看,支持的可以给个赞哇。

前言

在上一篇博客中,我们了解了strlen的模拟实现,以及冒泡排序,并且为了熟悉指针数组,我们还学习了用指针数组来模拟实现二维数组,本期博客将用其他指针类型来开篇,并学习指针和数组,函数之间的关系。 

目录

一、字符指针变量

 二、数组指针变量

三、二维数组传参

四、函数指针变量

五、typedef关键字

六、函数指针数组

七、转移表


一、字符指针变量

 什么是字符指针变量?

看下列代码:

#include<stdio.h>int main()
{char ch = 'c';char* pc = &ch;return 0;
}

上列代码中,pc就是字符指针变量,那如果后面的不是&ch,而是一个字符串又是如何存放在字符指针变量中的呢?

#include<stdio.h>int main()
{char* pc = "abcde";printf("%c ", *pc);return 0;
}

我们通过调试发现,最后的输出结果为a。所以字符串在存入指针变量的时候,是将首字符a的地址存入的。 

 接下来看以下代码:

#include<stdio.h>int main()
{char str1[] = "hello SILMY23";char str2[] = "hello SILMY23";const char* str3 = "hello SILMY23";const char* str4 = "hello SILMY23";if (str1 == str2)printf("same\n");elseprintf("%p\n%p\n",str1,str2);if (str3 == str4)printf("same\n");elseprintf("%p\n%p\n", str3, str4);return 0;
}

我们假设有两个字符数组,和两个用const修饰的字符指针变量,存放字符串hello SILMY23,我们看看在内存上它们是如何存放的?

运行结果如下:

 

我们发现在内存上,用const修饰的,它们是公用同一块空间的,也就是空间图如下所示:

 

用const修饰的常量字符串是不能被修改的,既然不能修改那么同样的内容就没有必要再开辟另外一个空间进行存放,所以两个str3和str4所指向的空间是相同的。 

 二、数组指针变量

我们在上一篇学习到,指针数组是用来存放指针的数组,那这一块数组指针,我们仍然采用类比的方式,字符指针,指向字符,是用来存放字符地址, 整型指针,指向整型,是用来存放整型地址,所以数组指针是指向数组,用来存放数组地址。例如:

int arr[10];
int (*p)[10] = &arr;

&arr拿到数组的地址。 p就是数组指针。

数组指针和指针数组

int* p[10];//指针数组
int (*p)[10];//数组指针

 字符数组指针:

char cha[8];
char (*pc)[8] = &cha;

所以数组指针类型:

int (*)[10]

char (*)[8] 

那数组指针如何初始化呢?就是用数组地址来初始化。

 那数组指针和指针数组的用法得分开

#include<stdio.h>int main()
{//指针数组int arr1[10] = { 1,2,3,4,5,6,7,8,9,10 };int* p[] = { arr1 };int i = 0;for (i = 0; i < 10; i++){printf("%d ", p[0][i]);}printf("\n");//数组指针int(*p1)[10] = &arr1;for (i = 0; i < 10; i++){printf("%d ", (*p)[i]);}return 0;
}

结果如下:

 

上述代码的分布图如下: 

总结:

指针数组是数组内的元素是一个个指针而数组指针是指向数组的指针

指针数组是数组,而数组指针就是个指针。

三、二维数组传参

我们在上一篇博客中,用指针数组模拟了二维数组,那二维数组传参的本质又是什么呢?

我们先来看二维数组的传参使用

#include<stdio.h>void print(int arr[2][5], int r, int c)
{int i = 0;for (i = 0; i < r; i++){int j = 0;for (j = 0; j < c; j++){printf("%d ", arr[i][j]);}printf("\n");}
}int main()
{int arr1[2][5] = { 1,2,3,4,5,6,7,8,9,10 };print(arr1, 2, 5);return 0;
}

二维数组的传参本质其实还是传了数组首元素的地址,传的是数组第一行第一列的地址,所以我们函数写形参还是可以写指针形式接收。

二维数组的每一行可以看作一个一维数组,这个一维数组可以看作是二维数组的一个元素,所以二维数组也可以认为是一维数组的数组,所以二维数组的首元素地址就是第一行的地址,也就是一个一维数组的地址。所以:

*(arr+i) == arr[i]

代码形参可以改造成这种,形参部分写成指向第一行的数组指针。而arr[i][j]==*(*(arr+i)+j)

#include<stdio.h>void print(int (*arr)[5], int r, int c)
{int i = 0;for (i = 0; i < r; i++){int j = 0;for (j = 0; j < c; j++){printf("%d ", *(*(arr + i) + j));}printf("\n");}
}int main()
{int arr1[2][5] = { 1,2,3,4,5,6,7,8,9,10 };print(arr1, 2, 5);return 0;
}

四、函数指针变量

我们已经接触过了很多指针变量,整型指针,数组指针,字符指针,那现在要新认识一个函数指,那顾名思义,函数指针,就是用来存放函数的地址,那函数名是否就是地址呢?

我们看之前写下的代码:

#include<stdio.h>void print(int (*arr)[5], int r, int c)
{int i = 0;for (i = 0; i < r; i++){int j = 0;for (j = 0; j < c; j++){printf("%d ", *(*(arr + i) + j));}printf("\n");}
}int main()
{int arr1[2][5] = { 1,2,3,4,5,6,7,8,9,10 };printf("%p ", print);printf("%p ", &print);return 0;
}

结果如下: 

 

我们发现,无论是函数名还是&函数名,都是函数的地址,而在数组当中,数组名和&数组名是不一样的。 

那现在要将print存入指针变量,在写法上还是类似数组指针的

那如何通过p调用print函数呢?

	(*p)(arr1, 2, 5);

首先是对p解引用获得函数地址,其次是传参。这样就可以通过函数指针调用函数了。

 但其实不写也是可以的。但写上(*)更容易理解。 

五、typedef关键字

typedef关键字是用来进行类型重命名的,可以将复杂的类型简单化:

//重命名数据类型
typedef unsigned int uint;
typedef double dl;
//重命名指针类型
typedef char* pcr;
typedef int* pint;
//重命名数组指针类型
typedef int(*parr_t)[5];
//重命名函数指针类型
typedef void(*pfun_t)(int);
//新的类型名必须在*的右边

六、函数指针数组

函数指针数组?说白了就是一个数组,这里面存放的都是函数指针的地址

在第四个知识点我们写了print函数,现在我们想把它放进一个函数指针数组里

void (*parr[1])(int, int, int) = { print };

函数指针类型是需要一样的,唯一不一样的是变量名我们给了一个数组来实现函数指针数组。 

七、转移表

函数指针数组的用途:转移表

普通四则计算器的实现:

#include <stdio.h>int Add(int x, int y)
{return x + y;
}int Sub(int x, int y)
{return x - y;
}int Mul(int x, int y)
{return x * y;
}int Div(int x, int y)
{return x / y;
}void menu()
{printf("***************************\n");printf("*****  1. add  2.sub  *****\n");printf("*****  3. mul  4.div  *****\n");printf("*****  0. exit        *****\n");printf("***************************\n");
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;int(*pfArr[5])(int, int) = { 0,Add,Sub,Mul,Div};do{menu();printf("请选择:>");scanf("%d", &input);if (input >= 1 && input <= 4){printf("请输入2个操作数:>");scanf("%d %d", &x, &y);ret = pfArr[input](x, y);printf("%d\n", ret);}else if(input == 0){printf("退出计算器\n");break;}else{printf("选择错误,请重新选择");}} while (input);return 0;
}

如果利用函数指针数组来简化代码,代码就会变得相对简短一些,增添一些功能,也只会从pfArr中添加。我们把这种用函数指针数组拿来做中间板的情况,就叫转移表。

感谢各位同伴的支持,本期指针进阶篇就讲解到这啦,如果你觉得写的不错的话,可以给个赞,若有不足,欢迎各位在评论区讨论。  

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

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

相关文章

年假作业11

一、选择题 ADDAADADC&#xff0c;BD,D,B,BD,D,C,CD 二、填空题 6 2&#xff0c;3,5,7,9 rgb *s, - a 2,5 *s 三、编程题 1、 #include <iostream> using namespace std; int main() {int arr[10]{10,20,30,40,50,60,70,80,90,100};int m;//从标准输入读取一个…

Javaweb之SpringBootWeb案例之AOP核心概念的详细解析

2.3 AOP核心概念 通过SpringAOP的快速入门&#xff0c;感受了一下AOP面向切面编程的开发方式。下面我们再来学习AOP当中涉及到的一些核心概念。 1. 连接点&#xff1a;JoinPoint&#xff0c;可以被AOP控制的方法&#xff08;暗含方法执行时的相关信息&#xff09; 连接点指的…

【Win10 触摸板】在插入鼠标时禁用触摸板,并在没有鼠标时自动启用触摸板。取消勾选连接鼠标时让触摸板保持打开状态,但拔掉鼠标后触摸板依旧不能使用

出现这种问题我的第一反应就是触摸板坏了&#xff0c;但是无意间我换了一个账户发现触摸板可以用&#xff0c;因此推断触摸板没有坏&#xff0c;是之前的账户问题&#xff0c;跟系统也没有关系&#xff0c;不需要重装系统。 解决办法&#xff1a;与鼠标虚拟设备有关 然后又从知…

Packet Tracer - Configuring ASA Basic Settings and Firewall Using CLI

Packet Tracer - 使用CLI配置ASA基本设置和防火墙 IP地址表 目标 验证连接并探索ASA设备使用CLI配置ASA的基本设置和接口安全级别使用CLI配置路由、地址转换和检查策略配置DHCP、AAA和SSH服务配置DMZ区域、静态NAT和访问控制列表&#xff08;ACL&#xff09; 场景 您的公司…

嵌入式STM32 单片机 GPIO 的工作原理详解

STM32的 GPIO 介绍 GPIO 是通用输入/输出端口的简称&#xff0c;是 STM32 可控制的引脚。GPIO 的引脚与外部硬件设备连接&#xff0c;可实现与外部通讯、控制外部硬件或者采集外部硬件数据的功能。 以 STM32F103ZET6 芯片为例子&#xff0c;该芯片共有 144 脚芯片&#xff0c…

Python eval函数

在Python编程中&#xff0c;eval()函数是一个强大且灵活的内置函数&#xff0c;用于动态执行字符串表达式或代码。尽管eval()函数具有强大的功能&#xff0c;但它也带来了一些潜在的安全风险&#xff0c;因此在使用时需要谨慎。本文将深入探讨eval()函数的用法、语法、示例代码…

LeetCode刷题计划---day2

07 #include <iostream> #include <iomanip> // 头文件用于控制输出格式 using namespace std;int main() {const int n 5; // 等级个数double grade[n] {4.0, 3.0, 2.0, 1.0, 0.0}; // 每个等级对应的分数string input;while (getline(cin, input)) { // 读入一…

AI - 碰撞避免算法分析(ORCA)

对比VO/RVO ORCA算法检测碰撞的原理和VO/RVO基本一样的&#xff0c;只是碰撞区域的计算去掉了一定时间以外才可能发生的碰撞&#xff0c;因此碰撞区域的扇形去掉了前面的部分&#xff0c;由圆锥头变成了个圆 另一个最主要的区别是&#xff0c;求新的速度&#xff0c;是根据相…

多进程面试题汇总

这里写目录标题 一、多进程1、进程的定义&#xff1a;2、单核多任务CPU执行原理3、进程的优点和缺点4、创建进程15、创建进程26、进程池6.1、进程池的作用6.2、原理图6.3、使用进程池的优点 7、进程间的通信&#xff08;Queue&#xff09;7.1、需求1&#xff1a;采用多进程将10…

GPIO八种工作模式

目录 一、推挽输出 二、开漏输出 三、复用推挽输出 四、复用开漏输出 五、浮空输入 六、上拉输入 七、下拉输入 八、模拟输入 GPIO八种配置模式&#xff0c;原理和使用场景&#xff0c;硬件原理如下图&#xff1a; 一、推挽输出 1、 原理 当控制栅极为低电平时&#x…

C++ STL: list使用及源码剖析

list使用 list常用函数及使用&#xff08;1&#xff09; #include <iostream> #include <list> #include <algorithm>int main() {// 创建liststd::list<int> myList {5, 2, 9, 1, 5, 6};// 打印liststd::cout << "Original list: &quo…

VS Code添加环境变量

有时候你会发现即使添加了环境变量, 打开VS Code的命令行终端也找不到对应的环境变量。遇到这种情况可以通过给VS Code的终端独立添加对应环境变量解决: 步骤1. 找到设置 步骤2. 找到windows终端环境变量配置 3. 在此处配置然后重新打开一个终端即可 (完)

ZISUOJ 2022年算法基础公选课练习四(Map)

说明&#xff1a; 博主为了提早预习数据结构和C的一些知识&#xff0c;自己琢磨外加查阅资料所写的代码&#xff0c;题目来源于22年初的学院老师组织的算法基础公选课的练习。我的代码甚至思路肯定存在许多不足和错误&#xff0c;欢迎大家批评指正。 题目列表&#xff1a; 问题…

HCIA-HarmonyOS设备开发认证V2.0-轻量系统内核基础-互斥锁mux

目录 一、互斥锁基本概念二、互斥锁运行机制三、互斥锁开发流程四、互斥锁使用说明五、互斥锁接口六、代码分析&#xff08;待续...&#xff09; 一、互斥锁基本概念 互斥锁又称互斥型信号量&#xff0c;是一种特殊的二值性信号量&#xff0c;用于实现对共享资源的独占式处理。…

STM32——OLED菜单(二级菜单)

文章目录 一.补充二. 二级菜单代码 简介&#xff1a;首先在我的51 I2C里面有OLED详细讲解&#xff0c;本期代码从51OLED基础上移植过来的&#xff0c;可以先看完那篇文章&#xff0c;在看这个&#xff0c;然后按键我是用的定时器扫描不会堵塞程序,可以翻开我的文章有单独的定时…

Flutter Android开发 梳理Google Material Design颜色体系

前言 做安卓开发&#xff08;Kotlin语言&#xff09;&#xff0c;Flutter开发的人员应该都听说过谷歌一直推崇的Material Design&#xff0c;而Material Design Color是其推崇的颜色体系&#xff0c;具体来说&#xff0c;Material Design Color是一套旨在帮助设计师和开发者创…

Panalog 日志审计系统 libres_syn_delete.php 前台RCE漏洞复现

0x01 产品简介 Panalog是一款日志审计系统,方便用户统一集中监控、管理在网的海量设备。 0x02 漏洞概述 Panalog日志审计系统 libres_syn_delete.php接口处存在远程命令执行漏洞,攻击者可执行任意命令,接管服务器权限。 0x03 影响范围 version <= MARS r10p1Free 0…

java+springboot+vue试题库在线学习系统05umj

技术路线&#xff1a; B/S架构&#xff0c;后端springboot框架&#xff0c;前端Vue.js框架。 主要功能模块&#xff08;至少六大功能&#xff09;&#xff0c;参考任务书并拓展 &#xff08;1&#xff09;用户管理模块&#xff1a;规定不同角色的用户对系统中各个功能模块的使用…

【经验】JLINK无法(单步)调试,JLINK固件的烧写

昨天终于准备开始进行S3C6410的裸机开发&#xff0c;写好了程序&#xff0c;编译生成了.axf文件&#xff0c;一切顺利的准备利用JLINK进行在线调试了&#xff0c;突然有种成功就在前面的感觉&#xff0c;Jlink也能被电脑正常的识别&#xff0c;利用AXD进行Jlink的相关设置也很正…