【C生万物】 指针篇 (进级) 下

news2025/1/19 11:16:04

 欢迎来到 Claffic 的博客 💞💞💞                               👉 专栏:《C生万物 | 先来学C》👈

前言:

承接上篇,这期继续C语言指针的学习。


目录

Part4:数组参数&指针参数

1.一维数组传参

2.二维数组传参

3.一级指针传参

4.二级指针传参

Part5:函数指针

1.引入

2.表示

Part6:函数指针数组

1.引入

2.表示

3.应用

Part7:回调函数

1.定义

2.应用


Part4:数组参数&指针参数

在敲代码时经常会有这样的情景:把【数组】或者【指针】传递给函数,那么函数该如何设计呢?这一部分会解答这个问题。 

1.一维数组传参

我将用这段代码来测试传参的正确姿势:

int main()
{
	int arr1[10] = { 0 };
	int* arr2[20] = { 0 };
	test1(arr1);
	test2(arr2);

	return 0;
}

看下列传参方式是否可行:

void test1(int arr[])
{}
// 可行:传数组,用数组接收,可以不指定大小
void test1(int arr[10])
{}
// 可行:传数组,用数组接收,大小保持一致
void test1(int* arr)
{}
// 可行:数组名代表首元素地址,用指针接收
void test2(int* arr[])
{}
// 可行:传递指针数组,用指针数组接收,可以不指定大小
void test2(int* arr[20])
{}
// 可行:传递指针数组,用指针数组接收,大小保持一致
void test2(int** arr)
{}
// 可行:数组名代表首元素地址,首元素类型是 int* ,地址类型是二级指针 int** 

总结:一维数组传参,要么用数组接收, 要么用指针接收

2.二维数组传参

测试代码:

int main()
{
	int arr[3][5] = { 0 };
	test(arr);
}

看下列传参方式是否可行:

void test(int arr[3][5])
{}
// 可行:相同形式接收
void test(int arr[][5])
{}
// 可行:指定列必须,行可以省略
void test(int arr[][])
{}
// 不可行:没有指定列
void test(int* arr)
{}
// 不可行:传递的二维数组,起码用首行的地址接收
void test(int* arr[5])
{}
// 不可行:形参表示指针数组,与实参数组的类型不匹配
void test(int(*arr)[5])
{}
// 可行:表示第一行的地址
void test(int** 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;
 int sz = sizeof(arr)/sizeof(arr[0]);
 //一级指针p,传给函数
 print(p, sz);
 return 0;
}

👁️‍🗨️输出结果: 

没意思吧?

那我们反过来想:当形参是一级指针的时候,函数能接收什么参数?

void test(int *p)
{}
// test函数能接收什么参数?
int a = 10;
int* p = &a;
int arr[10];
// 接收以下参数
test(&a);
test(p);
test(arr);

4.二级指针传参

同一级指针,二级指针传参,二级指针接收就行了。

例子: 

void test(int** ptr)
{
	printf("num = %d\n", **ptr);
}

int main()
{
	int n = 10;
	int* p = &n;
	int** pp = &p;
	test(pp);
	test(&p);
	return 0;
}

 👁️‍🗨️输出结果: 

反过来想: 当形参是二级指针的时候,函数能接收什么参数?

void test(int** pp)
{}
// 能接收什么参数?
int a = 10;
int* pa = &a;
int** ppa = &pa;
int* arr[10];
// 接收以下参数
test(&pa);
test(ppa);
test(arr);

Part5:函数指针

1.引入

既然指向单个变量的指针,有指向数组的指针,那么有没有指向函数的指针?

欸,还真有:

下面一段代码可以证明: 

void test()
{
	printf("Hello\n");
}

int main()
{
	printf("%p\n", test);
	printf("%p\n", &test);
	return 0;
}

👁️‍🗨️输出结果:

可见函数是有地址的,既然有地址,就可以存放起来作为指针;

并且:&函数名 与 函数名 都表示函数的地址。

2.表示

我先放出两种,你看看那种行得通:

// fun1和pfun2哪个有能力存放test函数的地址?
void (*pfun1)();
void *pfun2();

答:pfun1可以,( )内优先,* 与pfun1 先结合,说明 pfun1 是个指针,指向的是一个函数,指向的函数无参数,也无返回类型。

来个复杂的:

// 写出下列函数的函数指针
int Add(int x, int y)
{
	return x + y;
}

答: 

int (*pf)(int, int) = &Add;
// 开头是返回类型,最后的括号里是参数类型

函数指针的写法与数组指针的写法非常类似,可以类比着来。

使用:

int (*pf)(int, int) = &Add;
int ret = pf(2, 3);
int ret = (*pf)(2, 3);

两种使用方法均可。

Part6:函数指针数组

1.引入

我们已经学过了常量数组,指针数组,那么函数指针数组呢?

函数指针数组的解读就是:

一个数组,里面的元素类型为函数指针。

2.表示

知道了函数指针数组的含义,怎么表示呢? 

// 哪个是函数指针数组?
int (*parr1[10])();
int *parr2[10]();
int (*)() parr3[10];

答:parr1 是函数指针数组

解释:parr1 先与[ ]结合,说明 parr1 是数组,那数组的内容呢?
int (*)() 类型的函数指针。

3.应用

函数指针数组是有实际应用的,就是 转移表

说名字挺难理解的,这里举一个例子:

现在要你用C语言写一个计算器,菜单如下:

输入1是加法,输入2是减法,输入3是乘法,输入4是除法。

我们发现这种应用有一个特点:就是要调用多个不同的函数,

如果我们用 switch - case 语句,每个判断语句下调用相应的函数,那岂不是太挫了?

所以就要用到 转移表 ,即把 多个函数的指针存放到一个数组当中,需要就按下标访问调用即可。 

代码实现:

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 }; //定义函数指针数组,作转移表
	while (input)
	{
		printf("*************************\n");
		printf("*  1:add         2:sub  *\n");
		printf("*  3:mul         4:div  *\n");
		printf("*************************\n");
		printf("请选择:");
		scanf("%d", &input);
		if ((input <= 4 && input >= 1))
		{
			printf("输入操作数:");
			scanf("%d %d", &x, &y);
			ret = (*p[input])(x, y);
		}
		else
			printf("输入有误\n");
		printf("ret = %d\n", ret);
	}
	return 0;
}

是不是很方便呢?

Part7:回调函数

1.定义

回调函数就是一个通过函数指针调用的函数
如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数
回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。

简单来说,回调函数就是利用函数指针,实现函数调用函数的操作。

2.应用

那么回调函数是怎么应用的呢?

这里有个实例,就是冒泡排序模拟 qsort 函数,正好往期介绍过了,可以直接跳转:

冒泡排序模拟qsort函数 


总结: 

本篇是指针进级的最后一篇,到这里我相信你已经对指针有着很深刻的理解了,这么来看,指针还不是最困难的,对吧?

码文不易 

如果你觉得这篇文章还不错并且对你有帮助,不妨支持一波哦  💗💗💗

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

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

相关文章

股票量价关系基础知识8----图解各阶段量价关系:价平量增

图解各阶段量价关系&#xff1a;价平量增 价平量增是指在成交量增加的情况下&#xff0c;股价几乎维持在一个价位附近波动。 一、上涨初期的价平量增 &#xff08;一&#xff09;形态分析 在股价上涨的初期&#xff0c;价平量增是筹码良性换手的现象&#xff0c;这可能是主力在…

企业如何提高销售对CRM的使用率

CRM销售管理系统是帮助企业管理销售和客户的工具。它使企业能够跟踪和分析客户行为&#xff0c;管理客户关系&#xff0c;从而提高销售线索转化率。尽管CRM系统有着诸多的好处&#xff0c;但CRM的使用率往往很低&#xff0c;尤其是在销售团队中。为什么CRM使用率低销售不爱用&a…

gradle插件分享-手把手教你写gradle插件

gradle插件分享-手把手教你写gradle插件 写在前面&#xff1a; 在基础熟练的基础上&#xff0c;完全可以考虑基于Booster、ByteX等框架来开发&#xff0c;效率应该会高一些。 修改字节码的插件不止asm一个&#xff0c;还有javaassist等&#xff0c;可以多做一些尝试&#xff…

双令牌机制(chatgpt)

先记录下 双令牌机制主要用于增加Web应用程序的安全性。这种机制通常包括两种类型的令牌&#xff1a;访问令牌&#xff08;Access Token&#xff09;和刷新令牌&#xff08;Refresh Token&#xff09;。 1&#xff0e;访问令牌:访问令牌是用户完成身份验证后接收的令 牌&…

Three.js 模型体素化原理及实现

在本文中&#xff0c;我们探索了 3D 模型的体素化过程&#xff0c;重点是使用导入的 glTF 模型创建 3D 像素艺术。 本文包括一个最终演示&#xff0c;涵盖了可以使用体素化实现的各种 3D 效果。 我们将提供涵盖以下主题的分步指南&#xff1a; 确定 XYZ 坐标是否在 3D 网格内的…

SES2000浅地层剖面仪自带处理软件ISE2.95的处理步骤

SES2000是目前市面上主流浅地层剖面仪。它的自带处理软件ISE经常和设备一起更新&#xff0c;造成ISE版本众多&#xff0c;虽然数据采集的格式都是raw&#xff0c;但是低版本ISE软件打不开高版本raw数据&#xff0c;即使软件版本相近&#xff0c;比如都是2.95版本序列&#xff0…

AI测试|天猫精灵智能音箱测试策略与方法

一、业务介绍 2014年11月&#xff0c;亚马逊推出了一款全新概念的智能音箱&#xff1a;Echo&#xff0c;这款产品最大的亮点是将智能语音交互技术植入到传统音箱中&#xff0c;从而赋予了音箱人工智能的属性。这个被称为“Alexa”的语音助手可以像你的朋友一样与你交流&#x…

Grafana系列-统一展示-9-Jaeger数据源

系列文章 Grafana 系列文章 配置 Jaeger data source Grafana内置了对Jaeger的支持&#xff0c;它提供了开源的端到端分布式跟踪。本文解释了针对Jaeger数据源的配置和查询。 关键的配置如下: URL: Jaeger 实例的 URL, 如: http://localhost:16686 或 http://localhost:16…

PPT技能之新手入门,零基础光速进阶的宝藏

不会PPT只是借口&#xff0c;懒惰才是你的心里话。只要现在开始学习&#xff0c;不出三个月&#xff0c;华丽蜕变成PPT大神&#xff01;你的进步&#xff0c;我的功劳&#xff01; 你的关注&#xff0c;是我最大的动力&#xff01;你的转发&#xff0c;我的10W&#xff01;茫茫…

维京人的秘密:残暴背后的真相,敬畏神灵死后进入英灵殿

维京人&#xff0c;一个充满神秘色彩的名字&#xff0c;勾起了人们对于古代北欧残暴战士的想象。然而&#xff0c;维京人究竟是如何形成这样的形象&#xff0c;他们的传统和习俗又是如何塑造了他们的一生呢&#xff1f; 首先&#xff0c;我们要了解维京人的生活背景。维京人生活…

Linux线程同步(5)——互斥锁or自旋锁?

自旋锁概述 自旋锁与互斥锁很相似&#xff0c;从本质上说也是一把锁&#xff0c;在访问共享资源之前对自旋锁进行上锁&#xff0c;在访问完成后释放自旋锁&#xff08;解锁&#xff09;&#xff1b;事实上&#xff0c;从实现方式上来说&#xff0c;互斥锁是基于自旋锁…

shell脚本之“sort“、“uniq“、“tr“、“cut“、“split“、“paste“以及“eval“命令详解

文章目录 sort命令uniq命令tr命令cut命令split命令paste命令eval命令总结 sort命令 以行为单位对文件内容进行排序&#xff0c;也可以根据不同的数据类型来排序. 比较原则&#xff1a;从首字符向后&#xff0c;依次按ASCII码值进行比较&#xff0c;最后将他们按升序输出. 语法…

Chrome和edge报STATUS_STACK_BUFFER_OVERRUN错误的处理办法

Chrome和edge突然就报STATUS_STACK_BUFFER_OVERRUN错误&#xff0c;原因未知。 解决方案&#xff1a; Chrome 卸载本地的chrome访问https://www.chromedownloads.net/chrome64win/&#xff08;windows64&#xff09;https://www.chromedownloads.net/chrome32win/&#xff0…

母亲节到了,写一个简单的C++代码给老妈送上一个爱心祝福

&#x1f34e; 博客主页&#xff1a;&#x1f319;披星戴月的贾维斯 &#x1f34e; 欢迎关注&#xff1a;&#x1f44d;点赞&#x1f343;收藏&#x1f525;留言 &#x1f347;系列专栏&#xff1a;&#x1f319; C/C专栏 &#x1f319;请不要相信胜利就像山坡上的蒲公英一样唾…

快速上手Arthas

目录 基本概述 安装方式 基础指令 jvm相关指令 class/classloader相关指令 monitor/watch/trace相关指令 其他 基本概述 jconsole等工具都必须在服务端项目进程中配置相关的监控参数&#xff0c;然后工具通过远程连接到项目进程&#xff0c;获取相关的数据。这样就会带…

快速查询的秘籍——B+树索引

页和记录的关系示意图 InnoDB根据主键查找数据的过程是什么&#xff1f; 没有索引的查找是什么&#xff1f;索引查找和通过主键查找有什么关系&#xff1f; 索引是解决什么问题的&#xff1f; 索引是解决定位数据页的&#xff0c;而不是定位一个页中的数据的&#xff0c;定位…

MATLAB绘制动画(一)质点动画

vx 100*cos(1/3*pi); vy 100*sin(1/3*pi); t 0:0.005:18; x vx*t; y vy*-9.8*t.^2/2; comet(x,y) 这里只是截取了最后的画面&#xff0c;正常运行时&#xff0c;可以看到从最高点向下落的动作。 想要了解这段代码&#xff0c;我们要知道comet函数的意义 这个函数可以沿着…

ChatGPT 发布重磅更新,插件系统即将上线!

公众号关注 “GitHubDaily” 设为 “星标”&#xff0c;每天带你逛 GitHub&#xff01; 昨天凌晨&#xff0c;ChatGPT 为诸多 Plus 会员陆续开放了插件系统内测权限&#xff0c;申请比较早的用户&#xff0c;现在应该都能体验上最新的插件系统了。 为了让风暴来得更为猛烈&…

SQL在线刷题

牛客网学习SQL在线编程&#xff0c;牛客网在线编程&#xff0c;一共82道 用于实践的网站&#xff0c;在线运行SQL 目前43道&#xff0c;刷不动了&#xff0c;剩下的之后找机会搞 只记录有疑问的题目 简单 SQL196 查倒数第三 查找入职员工时间排名倒数第三的员工所有信息 …

js堆和栈

目录 关键句提取&#xff1a; 一、认识堆和栈 1、内存操作场景 2、数据结构场景 二、堆和栈的优缺点 1.栈(stack) 2.堆(heap) 3.总结&#xff1a; 三、堆和栈的溢出 四、 传值和传址 五、为什么会有栈内存和堆内存之分&#xff1f; 垃圾回收 标记清理 引用…