灵魂指针,教给(三)

欢迎来到白刘的领域   Miracle_86.-CSDN博客

系列专栏  C语言知识

先赞后看,已成习惯

   创作不易,多多支持!

目录

          一、 字符指针变量

二、数组指针变量

2.1 数组指针变量是什么

2.2 数组指针变量如何初始化

 三、二维数组传参本质

 四、函数指针变量

4.1 函数指针变量的创建

4.2 函数指针的使用

4.3 两段有意思的代码

4.3.1 typedef关键字

 五、函数指针数组

 六、转移表


一、 字符指针变量

指针的类型中有一种为字符指针char*

一般我们使用的时候,是这么使用的:

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

其实还有一种方法:

int main()
{const char* pstr = "hello bit.";//这⾥是把⼀个字符串放到pstr指针变量⾥了吗?printf("%s\n", pstr);return 0;
}

其实上述方法是将字符串的首字符的地址放在了指针pstr里了,也就是字符h的地址。

《剑指offer》中收录了这样一道题:

#include <stdio.h>
int main()
{char str1[] = "hello bit.";char str2[] = "hello bit.";const char* str3 = "hello bit.";const 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");return 0;
}

来看运行结果:

你猜对结果了嘛,下面我们来简单讲解一下这道题。

首先四个字符串,str1.2.3.4,在3和4的前面用const修饰了,所以这里的str3和str4是常量字符串。

而在C/C++中对于一个常量字符串,是不会额外开辟空间的,所以str3和str4的首地址是一个地址,所以两个字符串也是相同的。而str1和str2不是常量字符串,是可修改操作的,所以分别开辟了两个空间来存放这两个字符串,因此它俩的首字符地址也不相同,所以两个字符串也不相同。

二、数组指针变量

2.1 数组指针变量是什么

前面我们学习了一个类似,非常容易混淆的,叫指针数组,当时也挖坑了,说以后会讲这个。

灵魂指针,教给(二)-CSDN博客

 我们之前说,指针数组,它是数组,那数组指针呢?是指针

我们已经熟悉:

整型指针变量:int * pint; 用来存放整型变量的地址,能指向整型数据的指针。

 

浮点型指针变量:float * pint; 用来存放浮点型变量的地址,能指向浮点型数据的指针。

所以我们能推断出数组指针,它是用来存放数组地址,能指向数组的指针

来思考下面两行代码,分别是什么:

int *p1[10];
int (*p2)[10];

第一行是在前面我们学的指针数组,第二行为数组指针。

如何去记忆呢?我们要记住一个点就好,就是[ ]的优先级是比 * 高的,我们去看变量名和谁先结合就好,第一行先和中括号结合,所以是数组,变量名为数组名;第二行加了小括号,所以先和*结合,成了数组指针。

那数组指针的类型是什么呢?我们在学习数组的时候说去掉数组名,剩下的就是类型,那来看看数组指针的类型怎么写呢?

int (*)[10];

2.2 数组指针变量如何初始化

 数组指针用来存放数组的地址,那怎么获取数组的地址呢?没错,就是我们之前说的&arr。

int arr[10] = {0};
&arr;//得到的就是数组的地址

如果要存放整个数组的地址,就存放在数组指针中:

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

我们调试也能看到&arr和p的地址是一样的。

数组指针类型解析:

int (*p) [10] = &arr;|    |    ||    |    ||    |    p指向数组的元素个数|    p是数组指针变量名p指向的数组的元素类型

 三、二维数组传参本质

有了数组指针,我们就能详细了解二维数组传参本质了。

之前,我们将一个二维数组传给一个函数的时候,我们往往会这么写:

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

我们学完数组指针之后,还有其它的写法嘛?

我们再次理解一下二维数组,其实它就可以看成一个一维数组,只不过里面元素装的也是一维数组,所以二维数组的每一行都是一个数组。

所以,根据数组名是首元素地址这个规则,二维数组的数组名也就代表着第一行的地址,也就是一个一维数组的地址。根据上面的例子,第一行的数组的类型也就是int [5],所以第一行的地址类型就是int (*)[5]。那就意味着二维数组传参的本质也是传递了地址,传递的是第一行的一维数组的地址,那么形参也可以写成指针形式:

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

总结:二维数组传参,形参部分可以写成数组,也可以写成指针。

四、函数指针变量

4.1 函数指针变量的创建

通过类比关系,不难理解函数指针。它应该是存放指针地址的,可以通过函数地址调用函数的。

那函数指针到底是不是这样的呢?我们来做个测试:

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

运行结果如下:

我们确实打印出来了地址,所以函数确实是有地址的,并且函数名就是地址,所以可以用&函数名的方式获得函数的地址。

我们如果想将函数的地址存放起来,也非常简单,这就用到了函数指针变量,写法其实和数组指针的写法非常类似:

void test()
{printf("hehe\n");
}
void (*pf1)() = &test;
void (*pf2)() = test;
int Add(int x, int y)
{return x + y;
}
int(*pf3)(int, int) = Add;
int(*pf3)(int x, int y) = &Add;//x和y写上或者省略都是可以的

就是后面的改成参数了而已,非常简单。

函数指针类型详解:

int (*pf3) (int x, int y)|     |    ------------ |     |         ||     |         pf3指向函数的参数类型和个数的交代|     函数指针变量名pf3指向函数的返回类型int (*) (int x, int y) //pf3函数指针变量的类型

4.2 函数指针的使用

通过函数指针调用指针指向的函数。

#include <stdio.h>
int Add(int x, int y)
{return x + y;
}
int main()
{int(*pf3)(int, int) = Add;printf("%d\n", (*pf3)(2, 3));printf("%d\n", pf3(3, 5));return 0;
}

运行结果:

 

我们发现,写(*pf3)(2,3),或者pf3(3,5)都可以。

4.3 两段有意思的代码

代码1

 (*(void (*)())0)();
//首先void(*)()    --这是指针变量
//( void(*)() )0    --强制类型转换
//这就意味着我们假设0地址这个位置放着无参,返回类型为void的函数

代码2

void (*signal(int , void(*)(int)))(int);
//void(* signal(int,void(*)(int)))(int);
//声明了一个函数,signal(int,void(*)(int))
//它有两个参数,一个是int,一个是函数地址
//把signal的地址放进了void类型的函数指针,参数为int

这两段代码均出自于《C陷阱与缺陷这本书》。

4.3.1 typedef关键字

我们好久没讲关键字了,今天就新认识一个,叫typedef。它是用来干什么的呢?用来给类型重命名的,可以将复杂的类型简单化。

举个栗子:比如说你觉得unsigned int 写起来不方便,你想给它改成uint,代码如下:

typedef unsigned int uint;
//将unsigned int 重命名为uint

如果是指针类型,能否重命名呢?当然可以,比如将int*改成ptr_r:

typedef int* ptr_t;

但是对于数组指针和函数指针,稍微有点不同。

比如我们要将int(*arr)[5]改写成parr_t,我们就需要这么改:

typedef int(*parr_t)[5]; //新的类型名必须在*的右边

同理,函数指针也是这样,比如我们将void(*)(int)改成pf_t,就要这么写:

 typedef void(*pfun_t)(int);//新的类型名必须在*的右边

练练手,如果我们将刚刚的代码2进行简化,我们可以这么写:

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

五、函数指针数组

数组是一个存放相同类型数据的存储空间,我们已经学习了指针数组。

比如:

int *arr[10];
//数组的每个元素是int*

如果我们想把函数的地址都存储到一个数组中,那这个数组就叫函数指针数组,它是怎么定义的呢?

int (*parr1[3])();
int *parr2[3]();
int (*)() parr3[3];

答案是parr1,

parr1先和[3]结合,说明其是数组,然后数组的内容是什么呢?是int(*)( )类型的函数指针。

六、转移表

讲完了函数指针数组,我们来说一个听起来比较牛逼的东西——转移表。 虽然听起来很牛逼,但是它的本质就是我们刚刚学过的函数指针数组。

举个栗子,假如我们现在要写一个计算器的小程序,正常我们会这么写:

#include <stdio.h>
int add(int a, int b)
{return a + b;
}
int sub(int a, int b)
{return a - b;
}
int mul(int a, int b)
{return a * b;
}
int div(int a, int b)
{return a / b;
}
int main()
{int x, y;int input = 1;int ret = 0;do{printf("*************************\n");printf(" 1:add 2:sub \n");printf(" 3:mul 4:div \n");printf(" 0:exit \n");printf("*************************\n");printf("请选择:");scanf("%d", &input);switch (input){case 1:printf("输入操作数:");scanf("%d %d", &x, &y);ret = add(x, y);printf("ret = %d\n", ret);break;case 2:printf("输入操作数:");scanf("%d %d", &x, &y);ret = sub(x, y);printf("ret = %d\n", ret);break;case 3:printf("输入操作数:");scanf("%d %d", &x, &y);ret = mul(x, y);printf("ret = %d\n", ret);break;case 4:printf("输入操作数:");scanf("%d %d", &x, &y);ret = div(x, y);printf("ret = %d\n", ret);break;case 0:printf("退出程序\n");break;default:printf("选择错误\n");break;}} while (input);return 0;
}

很好想,但是这么写稍微有点冗余了(就是switch里有许多重复或者相似的代码片段)。

而我们学完了函数指针数组,我们就可以这么写,我们把加减乘除的函数都装到数组里,代码如下:

#include <stdio.h>
int add(int a, int b)
{return a + b;
}
int sub(int a, int b)
{return a - b;
}
int mul(int a, int b)
{return a * b;
}
int div(int a, int b)
{return a / b;
}
int main()
{int x, y;int input = 1;int ret = 0;int(*p[5])(int x, int y) = { 0, add, sub, mul, div }; //转移表do{printf("*************************\n");printf(" 1:add 2:sub \n");printf(" 3:mul 4:div \n");printf(" 0:exit \n");printf("*************************\n");printf("请选择:");scanf("%d", &input);if ((input <= 4 && input >= 1)){printf("输入操作数:");scanf("%d %d", &x, &y);ret = (*p[input])(x, y);printf("ret = %d\n", ret);}else if (input == 0){printf("退出计算器\n");}else{printf("输⼊有误\n");}} while (input);return 0;
}

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

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

相关文章

Redis + Caffeine = 王炸!!

在高性能的服务架构设计中,缓存是一个不可或缺的环节。在实际的项目中,我们通常会将一些热点数据存储到Redis或MemCache这类缓存中间件中,只有当缓存的访问没有命中时再查询数据库。在提升访问速度的同时,也能降低数据库的压力。 随着不断的发展,这一架构也产生了改进,在…

Selenium控制已运行的Edge和Chrome浏览器(详细启动步骤和bug记录)

文章目录 前期准备1. 浏览器开启远程控制指令&#xff08;1&#xff09;Edge&#xff08;2&#xff09;Chrome 2. 执行python代码&#xff08;1&#xff09;先启动浏览器后执行代码&#xff08;2&#xff09;通过代码启动浏览器 3. 爬取效果3. 完整代码共享3.1 包含Excel部分的…

【数据结构】初识二叉搜索树(Binary Search Tree)

文章目录 1. 二叉搜索树的概念2. 二叉搜索树的操作1.1 二叉搜索树的查找1.2 二叉搜索树的插入1.3 二叉搜索树的删除 1. 二叉搜索树的概念 二叉搜索树又称二叉排序树&#xff0c;它可能是一棵空树&#xff0c;也可能是具有以下性质的二叉树&#xff1a; 若它的左子树不为空&am…

力扣L5----- 58. 最后一个单词的长度(2024年3月11日)

1.题目 2.知识点 注1&#xff1a; lastIndexOf()它用于查找指定字符或子字符串在当前字符串中最后一次出现的位置。它的作用是从字符串的末尾向前搜索指定字符或子字符串&#xff0c;并返回其最后一次出现的位置的索引。 &#xff08;1&#xff09;例如&#xff0c;在字符串 …

Rust入门:C++和Rust动态库(dll)的相互调用

无论是C调用Rust动态库还是Rust调用C动态库&#xff0c;其操作基本都是一样地简单&#xff0c;基本和C调用C的动态库没什么区别&#xff0c;只需要列出所需要导入的函数&#xff0c;并链接到相应的lib文件即可。 这里&#xff0c;在windows中&#xff0c;我们以dll动态库为例说…

OpenCV学习笔记(一)——Anaconda下载和OpenCV的下载

OpenCV是图象识别中有巨大的应用场景&#xff0c;本篇文章以Python为基础。当初学OpenCV的时候&#xff0c;推使用在Anaconda编写代码&#xff0c;原因比较方便&#xff0c;下面我们对于Anaconda的下载过程进行演示。 Anaconda的下载 首先打开官网www.anaconda.com/download找…

数据结构---C语言栈队列

知识点&#xff1a; 栈&#xff1a; 只允许在一端进行插入或删除操作的线性表&#xff0c;先进后出LIFO 类似一摞书&#xff0c;按顺序拿&#xff0c;先放的书只能最后拿&#xff1b; 顺序栈&#xff1a;栈的顺序存储 typedef struct{Elemtype data[50];int top; }SqStack; SqS…

设计模式-行为型模式-模版方法模式

模板方法模式&#xff0c;定义一个操作中的算法的骨架&#xff0c;而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。[DP] 模板方法模式是通过把不变行为搬移到超类&#xff0c;去除子类中的重复代码来体现它的优势。 //首…

【Leetcode每日一题】 位运算 - 面试题 01.01. 判定字符是否唯一(难度⭐)(33)

1.题目解析 题目链接&#xff1a;面试题 01.01. 判定字符是否唯一 这个问题的理解其实相当简单&#xff0c;只需看一下示例&#xff0c;基本就能明白其含义了。 核心在于判断题目所给字符串是否存在相同字母&#xff0c;存在返回false即可&#xff0c;不存在返回true即可。 …

Caffeine缓存

本地缓存基于本地环境的内存&#xff0c;访问速度非常快&#xff0c;对于一些变更频率低、实时性要求低的数据&#xff0c;可以放在本地缓存中&#xff0c;提升访问速度 使用本地缓存能够减少和Redis类的远程缓存间的数据交互&#xff0c;减少网络 I/O 开销&#xff0c;降低这…

08 聚合函数

聚合函数 我们上一章讲到了 SQL 单行函数。实际上 SQL 函数还有一类&#xff0c;叫做聚合&#xff08;或聚集、分组&#xff09;函数&#xff0c;它是对一组数据进行汇总的函数&#xff0c;输入的是一组数据的集合&#xff0c;输出的是单个值。 1.聚合函数介绍 什么是聚合函…

for、while、do...while循环的使用

本篇文章只记录for、while、do...while循环的使用&#xff0c;由于java循环较为简单&#xff0c;所以直接上代码。 1、for循环 需求&#xff1a;循环遍历求和 1-100。 public class Demo {public static void main(String[] args) {int sum 0;for (int i 1; i < 100; i…

YOLOv5目标检测学习(4):YOLOV5源码的文件结构解析

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言①py、cpp、java后缀的文件②md、txt、yml后缀的文件③yaml后缀的文件 一、.github文件夹1.1 workflows文件夹&#xff1a;该文件夹通常包含GitHub Actions 的工…

RHEL9 DNF/YUM仓库管理软件包

DNF/YUM仓库管理软件包 一个基于RPM包的软件包管理器能够从指定的服务器自动下载RPM包并且安装&#xff0c;自动处理依赖性关系&#xff0c;并且一次性安装所有依赖的软件包C/S模式 Server服务端提供RPM软件包与数据库文件repodataClient客户端使用dnf仓库 常用组合 组合参…

解决Klipper下位机ID获取失败问题

使用硬件&#xff1a; 上位机&#xff1a;必趣派&#xff0c;版本CB1_Debian11_Klipper_kernel5.16_20230303 下位机&#xff1a;八爪鱼STM32F407 问题&#xff1a;上位机获取下位机ID失败。 解决&#xff1a;咨询DIY群友&#xff0c;也对这个问题不太了解。我调试过程中&…

Python 导入Excel三维坐标数据 生成三维曲面地形图(面) 4-1、线条平滑曲面(原始图形)

环境和包: 环境 python:python-3.12.0-amd64包: matplotlib 3.8.2 pandas 2.1.4 openpyxl 3.1.2 scipy 1.12.0 代码: import pandas as pd import matplotlib.pyplot as plt from mpl_toolkits.mplot3d import Axes3D from scipy.interpolate import griddata fro…

【vivado】 clock wizard 时钟IP

一、前言 MMCM和PLL是在FPGA设计中不可避免需要使用到的时钟资源&#xff0c;对于其功能及使用方法的理解是正确进行FPGA设计的前提。 二、Xilinx 时钟 IP配置 vivado中使用时钟向导(Clocking Wizard)配置时钟IP核&#xff0c;其框图如下&#xff1a; clk_in 输入时钟&#…

基于YOLOv8/YOLOv7/YOLOv6/YOLOv5的的商品标签识别系统(Python+PySide6界面+训练代码)

摘要&#xff1a;开发商品标签识别系统在提升零售业运营效率和顾客购物体验中发挥着关键作用。本篇博客详细阐述了如何应用深度学习技术构建此类系统&#xff0c;并提供了完整的代码实现。该系统基于高效的YOLOv8算法&#xff0c;并与YOLOv7、YOLOv6、YOLOv5进行了性能对比&…

*Javaweb -- MyBatis*

一:介绍: 1.MyBatis是一个优秀的 ①持久层 ②框架,用于简化JDBC的开发! ①:JAVAEE有三层的结构:表现层, 业务层, 持久层. 表现层代表的是页面的展示,业务层则指的是对于相关逻辑的处理, 而持久层, 指的则是对于数据进行持久化,保存在数据库当中. 持久层具体的来说就是负责…

数字孪生+工业互联网标识解析,打造智能工厂新标杆!

当前&#xff0c;工业4.0浪潮愈发澎湃&#xff0c;加快数字化、网络化、智能化发展成为了制造业转型升级的必然要求。 51WORLD基于数字孪生技术与工业互联网标识解析体系&#xff0c;打造了一个集协同化供应、个性化定制、智能化生产于一体的全连接产线孪生平台&#xff08;以…