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

深入理解指针:【探索指针的高级概念和应用二】

目录

一,数组参数、指针参数

1.一维数组传参

2.二维数组传参

3.一级指针传参

4.二级指针传参

二,函数指针

三,函数指针数组

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

四,指向函数指针数组的指针


一,数组参数、指针参数

我们在写代码的时候难免要把【数组】或者【指针】传给函数,那函数的参数该如何设计呢?下面我们就一起来探究一下。

1.一维数组传参

#include <stdio.h>
//数组传参,形参是可以写成数组形式的,因为这儿传参的本质是数组首元素的地址,所以大小可以不写
void test(int arr[])
{}//这儿不会去创建一个新的数组,这个大小是无意义的,所以数组里边的大小也可以省略
void test(int arr[10])
{}//数组传参的本质是传递数组首元素的地址;数组传参,形参也可以是指针
void test(int* arr)
{}//数组传参,形参用数组形式,数组的大小也可以省略
void test2(int* arr2[20])
{}//arr2的每个元素类型都是int*,这儿传过来的是首元素地址,即第一个元素的地址(int*的地址),
//所以就是将一级指针的地址取出来放在二级指针里边去
void test2(int** arr2)
{}int main()
{int arr[10] = { 0 };//定义了一个一维数组,数组里边有10个元素,每个元素是int类型int* arr2[20] = { 0 };//数组里边有20个元素,每个元素是int*类型test(arr);test2(arr2);
}

2.二维数组传参

//数组传参,形参的部分写成数组
void test(int arr[3][5])
{}//错误写法
//数组传参的时候,行可以省略,但是列绝对不能省略
void test(int arr[][])
{}//正确写法
void test(int arr[][5])
{}//错误写法
//数组名表示首元素的地址,即第一行的地址;而这是一个整型指针,
//整型指针是用来接受整型变量的地址的,所以这种写法是错误的
void test(int* arr)
{}//错误写法
//二维数组传过来拿指针数组接收了,应该用数组指针接收
void test(int* arr[5])
{}//正确写法
//这是一个数组指针,指向5个元素,每个元素是int类型,它可以指向数组中的第一、二、三行
void test(int(*arr)[5])
{}//错误写法
//二级指针是用来接收一级指针的地址的
void test(int** arr)
{}int main()
{int arr[3][5] = { 0 };//定义了一个三行五列的二维数组test(arr);//对数组进行传参
}

🍂总结:

二维数组传参,函数形参的设计只能省略第一个[ ]的数字。因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素,这样才方便运算。

3.一级指针传参

#include <stdio.h>
//一级指针传参的时候形参的部分写成一级指针就可以
void print(int* p, int sz)
{int i = 0;for (i = 0; i < sz; i++){printf("%d\n", *(p + i));//访问数组的每个元素}
}
int main()
{int arr[10] = { 1,2,3,4,5,6,7,8,9 };int* p = arr;//将数组的数组名赋给了p,本质是将数组首元素的地址赋给了pint sz = sizeof(arr) / sizeof(arr[0]);print(p, sz);//一级指针p,传给函数return 0;
}

🌴我们可以思考一下当一个函数的参数部分为一级指针的时候,函数能接收什么参数?

void test(int* p)
{}int a = 10;test(&a);//传整型变量的地址int* ptr = &a;
test(ptr);//传整型指针int arr[5];
test(arr);//传整型一维数组的数组名

4.二级指针传参

#include <stdio.h>
void test(int** ptr)//二级指针传过来拿二级指针接收
{printf("num = %d\n", **ptr);
}
int main()
{int n = 10;int* p = &n;int** pp = &p;test(pp);//把pp这个二级指针传给test函数test(&p);//p是一级指针变量,&p也是二级指针return 0;
}

🌴我们再思考一下当一个函数的参数部分为二级指针的时候,函数能接收什么参数? 

void test(int** p)
{}int main()
{int n = 10;int* p = &n;int** pp = &p;int* arr[6];test(&p);test(pp);test(arr);//数组名表示首元素的地址,就是int*的地址,传参后要用二级指针来接收return 0;
}

二,函数指针

我们知道数组指针指向数组的指针,存放的是数组的地址,&数组名就是数组的地址;

函数指针就是指向函数的指针,存放的是函数的地址,那怎么才能得到函数的地址呢?是不是&函数名呢?接下来我们通过一段代码来探究一下函数指针的神秘面纱:

int Add(int x, int y)
{return x + y;
}
int main()
{//&函数名就是函数的地址//函数名也是函数的地址printf("%p\n", &Add);printf("%p\n", Add);return 0;
}

🎈输出结果: 

我们可以看到输出的是两个相同的地址,而这两个地址都是Add函数的地址 ,所以&函数名函数名都能得到函数的地址。那我们的函数想要保存起来,该怎么做呢?看代码:

int Add(int x, int y)
{return x + y;
}
int main()
{int (*pf1)(int, int) = &Add;//pf1就是函数指针变量int ret = (*pf1)(3, 5);//通过函数指针变量找到函数地址并且调用它printf("%d\n", ret);return 0;
}

上面代码中的pf1是函数指针变量,它可以将我们的函数保存起来,pf1先和*结合,说明pf1是指针,指针指向的是一个函数,指向的函数有两个int类型的参数,返回值类型为int类型。


🍂 接下来我们再阅读两段有趣的代码(出自:《C陷阱和缺陷》):

🍃代码一:

int main()
{( *(void (*)())0 )();return 0;
}

在上面这段代码中我们从最熟悉的0开始下手,0是个数字;0前面的void (*)()这部分是指针指向了一个函数,函数没有参数,返回类型是void,所以这部分是一个函数指针类型;将类型放在括号里边就是要强制类型转换,转换完后前面加个*,就是要解引用,去调用这个函数,调用的这个函数也没有参数;

总结起来就是上面这段代码是在调用0地址处的函数,这个函数没有参数,返回类型是void。

🍃代码二:

int main()
{void (*signal(int, void(*)(int)))(int);return 0;
}

 上面这段代码比较复杂,我们可以将它简化为以下两部分

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

现在我们再来看这段代码就比较好分析了,首先它是一次函数声明,声明的是signal函数,signal 函数的参数有两个,第一个是int 类型,第二个是函数指针类型,该类型是void (*)(int),该函数指针指向的函数,参数是int,返回类型是void;signal函数的返回类型也是函数指针类型,该类型是void (*)(int),该函数指针指向的函数,参数是int,返回类型是void。

三,函数指针数组

通过前面的学习我们知道数组是一个存放相同类型数据的存储空间,例如指针数组:

int* arr[10];

 这是一个整型指针数组,存放的是整型指针,数组的每个元素是int*。

由上边的例子我们就可以知道函数指针数组也是一个数组,它存放的是函数指针(即函数的地址 ):

int (*parr1[10])();
//parr1 先和 [] 结合,说明 parr1是数组,
//数组的内容是int (*)() 类型的函数指针

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

🌴例子(计算器):

void menu()
{printf("************************\n");printf("***  1.Add    2.Sub  ***\n");printf("***  3.Mul    4.Div  ***\n");printf("***  0.exit          ***\n");printf("************************\n");
}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;
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();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;
}
  • 上面这段代码现在只具有加减乘除的功能,但是如果我想让它实现 a&&b、a||b、a&b、a|b、a>>b、a<<b等功能时,我们的运算会越来越多,菜单里边的功能相应的也会越来越多,同时类似加减乘除的函数也会越来越多,而且switch语句会越来越长,代码会冗余。
  • 那有没有什么办法让函数变得简洁呢,其实是有的。通过观察,会发现这些函数除了函数名和里边的计算不一样外,函数的参数都是两个int,返回类型都是int,所以我们可以通过函数指针数组来改写它。

🌴使用函数指针数组的实现:

void menu()
{printf("************************\n");printf("***  1.Add    2.Sub  ***\n");printf("***  3.Mul    4.Div  ***\n");printf("***  0.exit          ***\n");printf("************************\n");
}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;
}int main()
{int input = 0;int x = 0;int y = 0;int ret = 0;do{menu();printf("请选择:> ");scanf("%d", &input);//函数指针数组 - 转移表int (*pfarr[5])(int, int) = { NULL,Add,Sub,Mul,Div };//放个NULL的原因是将这些操作函数的下标往右挤一位if (0 == input){printf("退出计算器\n");}else if (input >= 1 && input <= 4){printf("请输入操作数:>");scanf("%d%d", &x, &y);ret = (pfarr[input])(x, y);printf("ret= %d\n", ret);}else{printf("选择错误,请重新选择!\n");}} while (input);return 0;
}

四,指向函数指针数组的指针

🎈我们先来看一下整型指针数组:

int a = 10;
int b = 20;
int c = 30;//整型指针数组,数组的每个元素是int*类型
int* arr[] = { &a, &b, &c };//p是指针,是指向整型指针数组的指针
int* (*p)[3] = &arr;

有了上面的例子我们再来看指向函数指针数组的指针:

指向函数指针数组的指针是一个 指针
指针指向一个 数组 ,数组的元素都是 函数指针

void test(const char* str)
{printf("%s\n", str);
}
int main()
{//函数指针pfunvoid (*pfun)(const char*) = test;//函数指针的数组pfunArrvoid (*pfunArr[5])(const char* str);pfunArr[0] = test;//指向函数指针数组pfunArr的指针ppfunArrvoid (*(*ppfunArr)[5])(const char*) = &pfunArr;return 0;
}

相关文章:

深入理解指针:【探索指针的高级概念和应用二】

目录 一&#xff0c;数组参数、指针参数 1.一维数组传参 2.二维数组传参 3.一级指针传参 4.二级指针传参 二&#xff0c;函数指针 三&#xff0c;函数指针数组 &#x1f342;函数指针数组的用途&#xff08;转移表&#xff09;&#xff1a; 四&#xff0c;指向函数指针…...

腾讯觅影数智医疗影像平台获颁世界互联网领先科技成果大奖

11月8日&#xff0c;2023年世界互联网大会乌镇峰会在乌镇举行&#xff0c;腾讯再度获颁“世界互联网领先科技成果”大奖。腾讯健康总裁吴文达在世界互联网领先科技成果发布活动中介绍&#xff0c;“腾讯觅影数智医疗影像平台”已全面开放20多个医疗AI引擎助力科研创新&#xff…...

鸿蒙开发工具DevEco Studio的下载和安装

一、DevEco Studio概述 1、简介 HUAWEI DevEco Studio&#xff08;获取工具请单击链接下载&#xff0c;以下简称DevEco Studio&#xff09;是基于IntelliJ IDEA Community开源版本打造&#xff0c;为运行在HarmonyOS和OpenHarmony系统上的应用和服务&#xff08;以下简称应用…...

【原理篇】四、自定义starter

文章目录 1、案例分析2、业务功能的实现3、中途调试4、开启定时任务打印报表5、引入属性配置类&#xff0c;写活业务参数配置6、拦截器7、开启yml提示功能 做一个记录系统访客独立IP访问次数的功能&#xff0c;并把它自定义成一个starter&#xff0c;实现&#xff1a;在现有项目…...

redisTemplate不支持zpopmax,解决方案使用reverseRangeWithScore

在redis客户端可以使用zpopmax redisTemplate不支持zpopmax 解决方案 使用reverseRangeWithScore 接下来我们进行测试 我们要返回最大的value&#xff0c;应该是c import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.beans.factory.a…...

基于深度模型的日志异常检测

本文转载自&#xff1a;深度模型的日志异常检测&#xff0c;还有谁不会&#xff1f;PPT下载地址&#xff1a;https://bbs.huaweicloud.com/forum/thread-100052-1-1.html视频讲解地址&#xff1a;https://bbs.huaweicloud.com/live/DevRun_live/202101222000.html 文章目录 1. …...

最大连续子数组

最大连续子数组&#xff08;Maximum Subarray&#xff09;问题是一个经典的算法问题&#xff0c;其目标是在给定的整数数组中找到一个连续的子数组&#xff0c;使得该子数组的元素之和最大。这个问题有多种解决方法&#xff0c;其中包括暴力解法、分治法和动态规划等。 下面是…...

【FastCAE源码阅读5】使用VTK实现鼠标拾取对象并高亮

鼠标拾取对象是很多软件的基本功能。FastCAE的拾取比较简单&#xff0c;是通过VTK实现的。 对几何而言&#xff0c;拾取类型切换在工具栏上&#xff0c;单击后再来单击视图区对象进行拾取&#xff0c;拾取后的对象会高亮显示。效果如下图&#xff1a; 一、拾取对象 拾取对象…...

【全志H616 使用标准库 完成自制串口库(分文件实现) orangepi zero2(开源)】.md updata: 23/11/07

文章目录 H616 把玩注意&#xff1a;Linux内核版本5.16 及以上&#xff0c;需手动配置i2c-3 uart5驱动配置示例 分文件编译时需将每个文件一同编译 &#xff08;空格隔开&#xff09;例&#xff1a; ggc a.c b.c b.h -lpthread -lxxx..; 常用命令查看驱动文件查看内核检测信息/…...

小白学爬虫:手机app分享商品短连接获取淘宝商品链接接口|淘宝淘口令接口|淘宝真实商品链接接口|淘宝商品详情接口

通过手机APP分享的商品短链接&#xff0c;我们可以调用相应的接口来获取淘口令真实URL&#xff0c;进而获取到PC端的商品链接及商品ID。具体步骤如下&#xff1a; 1、通过手机APP分享至PC端的短链接&#xff0c;调用“item_password”接口。 2、该接口将返回淘口令真实URL。 3…...

python 应用之 request 请求调用

场景&#xff1a; 验证一个第三方接口 目录 一、应用实例 1、预准备工作 1&#xff09;、引用包 2&#xff09;、生成随机串 3&#xff09;、获得当前时间戳 4&#xff09;、HASH 5&#xff09;、header处理 6&#xff09;、请求处理 2、requests请求 1&#xff09…...

BeanUtils.copyProperties浅拷贝的坑你得知道?

今天想写一篇文章&#xff0c;主要关于深拷贝和浅拷贝相关的&#xff0c;主要是最近写代码的时候遇到一个BUG&#xff0c;刚好涉及到浅拷贝导致的问题。 问题背景 现在有一个需要是需要修改门店信息&#xff0c;门店也区分父门店和子门店&#xff0c;父门店被编辑更新是需要通过…...

ubuntu安装rabbitMQ 并 开启记录消息的日志

apt-get update apt-get install rabbitmq-server rabbitmqctl add_user root password // 设置用户名密码 rabbitmqctl set_user_tags root administrator // 设置为管理员身份 rabbitmqctl set_permissions -p / root ".*" ".*" ".*" //为…...

思维模型 首因效应

本系列文章 主要是 分享 思维模型&#xff0c;涉及各个领域&#xff0c;重在提升认知。先入为主&#xff0c;一见钟情。 1 首因效应的应用 1.1 面试中的首因效应 小李是一名应届毕业生&#xff0c;他准备参加一家知名互联网公司的面试。在面试前&#xff0c;他做了充分的准备…...

Redis极速上手开发手册【Redis全面复习】

文章目录 什么是RedisRedis的特点Redis的应用场景Redis安装部署Redis基础命令Redis多数据库特性Redis数据类型Redis数据类型之stringRedis数据类型之hashRedis数据类型之listRedis数据类型之setRedis数据类型之sorted set案例&#xff1a;存储高一班的学员信息 Redis封装工具类…...

[动态规划] (十四) 简单多状态 LeetCode LCR 091.粉刷房子

[动态规划] (十四) 简单多状态 LeetCode LCR 091.粉刷房子 文章目录 [动态规划] (十四) 简单多状态 LeetCode LCR 091.粉刷房子题目解析解题思路状态表示状态转移方程初始化和填表顺序返回值 代码实现总结 LCR 091. 粉刷房子 题目解析 (1) 一排房子&#xff0c;共有n个 (2) 染…...

【VSS版本控制工具】

VSS版本控制工具 1 安装 VSS2 服务器端配置3 新建用户4 客户端配置Vss2005Vs20055 客户端详细操作 1 安装 VSS 第一步&#xff1a;将VisualSourceSafe2005安装包解压。 第二步&#xff1a;找到setup.exe双击运行。 第三步&#xff1a;在弹出的界面复选框中选中Iaccepttheterms…...

数据持久化技术(Python)的使用

传统数据库连接方式&#xff1a;mysql&#xff08;PyMySQL&#xff09;ORM 模型&#xff1a;SQLAlchemy MyBatis、 Hibernate PyMySQL 安装&#xff1a; pip install pymysql简单使用 利用 pymysql.connect 建立数据库连接并执行 SQL 命令&#xff08;需要提前搭建好数据库…...

第23章(上)_索引原理之索引与约束

文章目录 索引索引分类主键选择索引的代价 约束外键约束约束与索引的区别 索引使用场景不要使用索引的场景总结 索引 索引的概念&#xff1a;索引是一种有序的存储结构。索引按照单个或多个列的值进行排序。 索引的目的&#xff1a;提升搜索效率。 索引分类 按照数据结构分为…...

金蝶云星空BOS设计器中基础资料字段属性“过滤”设置获取当前界面的基础资料值作为查询条件

文章目录 金蝶云星空BOS设计器中基础资料字段属性“过滤”设置获取当前界面的基础资料值作为查询条件背景说明业务需求格式BOS配置 金蝶云星空BOS设计器中基础资料字段属性“过滤”设置获取当前界面的基础资料值作为查询条件 背景说明 序列号档案是基础资料&#xff0c;资料里…...

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…...

linux 错误码总结

1,错误码的概念与作用 在Linux系统中,错误码是系统调用或库函数在执行失败时返回的特定数值,用于指示具体的错误类型。这些错误码通过全局变量errno来存储和传递,errno由操作系统维护,保存最近一次发生的错误信息。值得注意的是,errno的值在每次系统调用或函数调用失败时…...

基于Docker Compose部署Java微服务项目

一. 创建根项目 根项目&#xff08;父项目&#xff09;主要用于依赖管理 一些需要注意的点&#xff1a; 打包方式需要为 pom<modules>里需要注册子模块不要引入maven的打包插件&#xff0c;否则打包时会出问题 <?xml version"1.0" encoding"UTF-8…...

[Java恶补day16] 238.除自身以外数组的乘积

给你一个整数数组 nums&#xff0c;返回 数组 answer &#xff0c;其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&#xff0c;且在 O(n) 时间复杂度…...

select、poll、epoll 与 Reactor 模式

在高并发网络编程领域&#xff0c;高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表&#xff0c;以及基于它们实现的 Reactor 模式&#xff0c;为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。​ 一、I…...

让回归模型不再被异常值“带跑偏“,MSE和Cauchy损失函数在噪声数据环境下的实战对比

在机器学习的回归分析中&#xff0c;损失函数的选择对模型性能具有决定性影响。均方误差&#xff08;MSE&#xff09;作为经典的损失函数&#xff0c;在处理干净数据时表现优异&#xff0c;但在面对包含异常值的噪声数据时&#xff0c;其对大误差的二次惩罚机制往往导致模型参数…...

Python 实现 Web 静态服务器(HTTP 协议)

目录 一、在本地启动 HTTP 服务器1. Windows 下安装 node.js1&#xff09;下载安装包2&#xff09;配置环境变量3&#xff09;安装镜像4&#xff09;node.js 的常用命令 2. 安装 http-server 服务3. 使用 http-server 开启服务1&#xff09;使用 http-server2&#xff09;详解 …...

SpringAI实战:ChatModel智能对话全解

一、引言&#xff1a;Spring AI 与 Chat Model 的核心价值 &#x1f680; 在 Java 生态中集成大模型能力&#xff0c;Spring AI 提供了高效的解决方案 &#x1f916;。其中 Chat Model 作为核心交互组件&#xff0c;通过标准化接口简化了与大语言模型&#xff08;LLM&#xff0…...

Tauri2学习笔记

教程地址&#xff1a;https://www.bilibili.com/video/BV1Ca411N7mF?spm_id_from333.788.player.switch&vd_source707ec8983cc32e6e065d5496a7f79ee6 官方指引&#xff1a;https://tauri.app/zh-cn/start/ 目前Tauri2的教程视频不多&#xff0c;我按照Tauri1的教程来学习&…...

Python异步编程:深入理解协程的原理与实践指南

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 持续学习&#xff0c;不断…...