前言

大家好,这里是 Hello_Embed。在前一篇笔记中,我们用循环实现了 LED 闪烁,其中重复使用了两段几乎一样的延时代码:

for(i = 0; i < 100000000; i++);  // 延时

这种重复不仅让代码冗余,还不利于后续修改(比如想调整延时时间,需要改两处)。有没有办法把这段延时 “打包” 成一个可复用的模块?这就需要用到 C 语言中的函数。本文将从函数的定义、声明讲起,结合实例说明参数传递的两种方式(值传递与地址传递),以及函数在嵌入式开发中的实用价值。

📦 函数的基本概念:封装重复代码

函数的核心作用是 “封装重复逻辑”,让代码更简洁、易维护。以延时功能为例,我们可以把重复的 for 循环封装成一个函数,需要时直接调用。

函数的定义与声明

函数的定义格式为:

返回类型 函数名(参数列表) {// 函数体:实现具体功能
}
  • 返回类型:表示函数执行后返回的数据类型(若无需返回值,用void);
  • 参数列表:函数接收的输入(若无需参数,用void);
  • 函数体:实现功能的代码块。
无参数、无返回值的函数

将延时功能封装成无参数函数

void delay(void)  // 第一个void:无返回值;第二个void:无参数
{int i;for(i = 0; i < 100000000; i++);  // 延时逻辑
}

调用时直接写delay();即可,替代原来的 for 循环。这样修改延时时间只需改函数内部,无需到处找重复代码。如果此函数涉及公司机密,还可以将函数封装为库,别人使用直接调用库就可以了。

带参数的函数

如果想灵活控制延时时长,可以给函数添加参数:

void delay(int cnt)  // 参数cnt:控制循环次数
{int i;for(i = 0; i < cnt; i++);  // 用参数cnt替代固定值
}

调用时通过delay(100000000);指定延时长度,甚至可以根据需求动态调整(比如delay(50000000);实现更短的延时)。

带返回值的函数

有的时候我们可能会遇到有着两个不同参数的函数,如下:

int add(int a, int b)
{int sum = a + b;return sum;
}

此时函数就有返回值sum,所以add的返回类型是int。这段代码还可以将sum省去,直接返回a + b。下面我们来实战演练一遍:

#include <stdio.h>
int add(int a, int b) { return a + b; }int main()
{int a1 = 10, b1 = 20;// 方式1:将返回值赋给变量,再使用int c1 = add(a1, b1);printf("The sum of %d and %d is %d\n", a1, b1, c1);// 方式2:直接在表达式中使用返回值printf("The sum of %d and %d is %d\n", a1, b1, add(a1, b1));return 0;
}

运行结果如图:
请添加图片描述
可以看到我们有两种方法输出两数之和:一种是将函数值赋给定义好的c1,再打印c1;另一种是直接打印函数返回的两数之和的值,这种使用方式也是合规的。

🔄 参数传递的两种方式

函数参数的传递方式决定了函数内部能否修改外部变量,这在嵌入式开发中尤为重要(比如通过函数修改硬件寄存器的值)。

值传递:仅传递变量的 “副本”

先看一个例子:

#include <stdio.h>
// 尝试修改参数a的值
void change_val(int a)
{printf("The value of a 1 is %d\n", a);  // 打印传入的值a = 200;  // 修改参数aprintf("The value of a 2 is %d\n", a);  // 打印修改后的值
}int main()
{int a = 100;change_val(a);  // 传入a的值printf("The value of a 3 is %d\n", a);  // 打印main中的areturn 0;
}

运行结果如图,函数内部的a被改成了 200,但 main 中的a仍为 100:
请添加图片描述
我们用反证法来理解这段代码的结果:假设main中的achange_val中的a用同一块内存,顺序执行时,先是a = 100,接着调用函数打印一次,此时打印出a为 100;接着a = 200再打印,a应为 200;函数执行完毕后在主程序里再次打印,理论上a应为 200,但现实结果却是 100。这说明:两个变量a所占内存不同
为什么会这样?这涉及到 “全局变量”“局部变量” 和 “栈” 的概念:

  • 在函数之外定义的变量称为全局变量,在函数之内定义的变量称为局部变量。
  • 程序运行时,内存中会划出一块区域称为 “栈”,还有一个硬件寄存器叫SP(栈指针)。当运行到main函数时,会为其分配空间,SP会从原来的位置往下移动,SP原来位置与现在位置之间的区域就是main的栈,里面存放main的局部变量。
  • 同样,change_val函数被调用时,也会有自己的栈来存放它的局部变量。
    所以,代码中同名的a实际上是不同的变量:main中的amain的栈里,change_val中的a在它自己的栈里。这段代码所侧重的恰恰是参数之间的传递:我们将maina的值 100 传入change_val中,所以第一次打印 100;函数内修改的是自己栈里的a,所以第二次打印 200;最后打印的是main函数中定义的a,所以还是 100。
地址传递:通过指针修改原变量

那如果我就是想通过change_val中的a改变main中的a的值呢?当然有办法解决,这个时候就需要请出指针了:

#include <stdio.h>
// 用指针接收地址,修改原变量
void change_val(int *a)  // 参数为int型指针
{printf("The value of a 1 is %d\n", *a);  // 访问指针指向的变量*a = 200;  // 修改指针指向的变量(即main中的a)printf("The value of a 2 is %d\n", *a);
}int main()
{int a = 100;change_val(&a);  // 传入a的地址printf("The value of a 3 is %d\n", a);  // main中的a被修改return 0;
}

运行结果如图,main 中的a成功被改为 200:
请添加图片描述
原理:指针a存储了 main 中a的地址,通过*a可以直接操作原变量的内存,因此修改会生效。这种方式在嵌入式中常用(比如通过函数修改寄存器的值,关于指针的更多知识可参考之前的笔记)。当change_val函数执行完毕,SP会重新指回main函数,代表一个函数的结束。

结尾

通过这篇笔记,我们系统学习了函数的基本概念:从无参数、无返回值的简单封装(如延时函数),到带参数、带返回值的灵活应用(如加法函数),再到参数传递的两种方式 —— 值传递(仅传副本,不影响原变量)和地址传递(用指针修改原变量)。我们还理解了局部变量在栈中的存储特点,这是搞懂参数传递机制的关键。
函数是 C 语言模块化编程的核心,在嵌入式开发中,我们可以将硬件初始化、数据处理等逻辑封装成函数,让代码更清晰、易复用。下一篇笔记,我们会进一步学习函数的递归。Hello_Embed 会继续陪伴大家,一步步掌握嵌入式 C 语言的精髓,敬请期待~

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

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

相关文章

第一个大语言模型的微调

模型推理 现在,我们的模型应该能够针对输入的任何短句生成类似尤达大师风格的句子作为回应。 该模型要求其输入格式规范。我们需要构建一个 “消息” 列表 —— 在这个案例中,就是来自用户的消息 —— 并通过提示表明轮到模型进行输出,以促使其做出回答。 add_generation…

Linux内核驱动开发核心问题全解

&#x1f4d6; 推荐阅读&#xff1a;《Yocto项目实战教程:高效定制嵌入式Linux系统》 &#x1f3a5; 更多学习视频请关注 B 站&#xff1a;嵌入式Jerry Linux内核驱动开发核心问题全解 本文系统梳理了 Linux 驱动开发、内核同步、中断处理、内存管理、进程通信、系统启动等典型…

【C++篇】C++11入门:踏入C++新世界的大门

文章目录C11简介列表初始化1. {}初始化2. initializer_list容器initializer_list的使用场景声明1. auto2. decltype3. nullptrSTL中的变化1. 新容器array容器forward_list容器unordered_map和unordered_set容器2. 新接口C11简介 C98/03&#xff1a;在2003年C标准委员会曾经提交…

Java 日期时间处理:分类、用途与性能分析

Java提供了多种日期时间处理API&#xff0c;随着版本演进不断改进。以下是主要日期时间类的分类、用途和性能分析&#xff1a;一、Java日期时间API分类1. 传统日期时间API (Java 1.0/1.1)java.util.Date - 表示特定的瞬间&#xff0c;精确到毫秒java.util.Calendar - 抽象类&am…

[Linux]学习笔记系列 --GCC

文章目录属性__cleanup__attribute_malloc__ 用于标记函数返回一个新分配的内存块__attribute_alloc_size__ 用于指定分配的内存大小__attribute__((const)) 标记为纯函数(pure function)__attribute__((__externally_visible__)) 使其在编译器优化过程中保持对外部模块的可见性…

【龙泽科技】汽车维护与底盘拆装检修仿真教学软件【风光580】

产品简介汽车维护与底盘拆装检修仿真教学软件是依托《全国职业院校技能大赛》“汽车维修”赛项中“汽车维护与底盘拆装检修模块”竞赛模块&#xff0c;自主开发的一款仿真教学软件。软件采用仿真仿真技术模拟实际汽车维修工的岗位技能操作流程&#xff0c;操作内容主要包括&…

Spring之【循环引用】

目录前置知识SingletonBeanRegistryDefaultSingletonBeanRegistrySpring中处理循环引用的流程分析定义两个具有循环引用特点的Bean执行A的实例化执行A的属性填充(执行过程中发现A依赖B&#xff0c;就去执行B的实例化逻辑)执行B的实例化执行B的属性填充执行B的初始化执行A的属性…

LRU缓存淘汰算法的详细介绍与具体实现

LRU&#xff08;Least Recently Used&#xff0c;最近最少使用&#xff09;是一种基于时间局部性原理的缓存淘汰策略。其核心思想是&#xff1a;最近被访问的数据在未来更可能被再次使用&#xff0c;而最久未被访问的数据应优先被淘汰&#xff0c;从而在有限的缓存空间内保留高…

JS-第十九天-事件(一)

一、事件基础概念1.1 事件三要素事件源&#xff1a;触发事件的元素事件类型&#xff1a;事件的种类&#xff08;如click、mouseover等&#xff09;事件处理程序&#xff1a;响应事件的函数1.2 事件流机制事件传播分为三个阶段&#xff1a;捕获阶段&#xff1a;事件从顶层开始&a…

Matplotlib(三)- 图表辅助元素

文章目录一、图表辅助元素简介二、坐标轴的标签、刻度范围和刻度标签1. 坐标轴标签1.1 x轴标签1.2 y轴标签1.3 示例&#xff1a;绘制天气气温折线图2. 刻度范围和刻度标签2.1 刻度范围2.1.1 x轴刻度范围2.1.2 y轴刻度范围2.2 刻度标签2.2.1 x轴刻度标签2.2.2 y轴刻度标签2.3 示…

【Linux基础知识系列】第七十八篇 - 初识Nmap:网络扫描工具

在网络管理和安全领域&#xff0c;网络扫描是一个不可或缺的工具。它可以帮助网络管理员了解网络中的设备、服务以及潜在的安全漏洞。Nmap&#xff08;Network Mapper&#xff09;是一个功能强大的开源网络扫描工具&#xff0c;它能够快速发现网络中的主机、端口和服务&#xf…

EasyGBS的两种录像回看

EasyGBS 支持两种录像回看&#xff0c;即“平台端”的录像回看和“设备端”的录像回看。本期我们来介绍两者的区别和使用方法。一、平台端录像1、什么是平台端录像平台端录像是指由 EasyGBS 平台直接录制并存储。2、配置平台端录像进入平台&#xff0c;依次点击【录像回放】→【…

大模型学习思路推荐!

为进一步贯彻落实中共中央印发《关于深化人才发展体制机制改革的意见》和国务院印发《关于“十四五”数字经济发展规划》等有关工作的部署要求&#xff0c;深入实施人才强国战略和创新驱动发展战略&#xff0c;加强全国数字化人才队伍建设&#xff0c;持续推进人工智能从业人员…

数据库连接池性能优化实战

背景我们公司正在处于某个项目的维护阶段&#xff0c;领导对资源告警比较重视&#xff0c;服务器资源告警的就不说了&#xff0c;运维同学每隔一小时都会检测线上环境的应用服务信息&#xff0c;例如&#xff1a;网关日志响应时间告警/nginx日志接口响应时间告警/日志关键字异常…

Excel常用函数大全,非常实用

一、数学与统计函数1. SUM作用&#xff1a;求和SUM(number1, [number2], ...)SUM(A1:A10) ➔ 计算A1到A10单元格的总和注意&#xff1a;自动忽略文本和空单元格2. AVERAGE作用&#xff1a;计算平均值AVERAGE(number1, [number2], ...)AVERAGE(B2:B20) ➔ 计算B列20个数据的平均…

性能优化(一):时间分片(Time Slicing):让你的应用在高负载下“永不卡顿”的秘密

性能优化(一)&#xff1a;时间分片&#xff08;Time Slicing&#xff09;&#xff1a;让你的应用在高负载下“永不卡顿”的秘密 引子&#xff1a;那张让你浏览器崩溃的“无限列表” 想象一个场景&#xff1a;你需要渲染一个包含一万个项目的列表。在我们的“看不见”的应用中&a…

《C++》STL--list容器详解

在 C 标准模板库(STL)中&#xff0c;list 是一个非常重要的序列容器&#xff0c;它实现了双向链表的数据结构。与 vector 和 deque 不同&#xff0c;list 提供了高效的插入和删除操作&#xff0c;特别是在任意位置。本文将深入探讨 list 容器的特性、使用方法以及常见操作。 文…

Day 28:类的定义和方法

DAY 28 类的定义和方法 知识点学习 1. 类的定义 在Python中&#xff0c;类是创建对象的模板。使用class关键字来定义一个类。类名通常采用首字母大写的命名方式&#xff08;PascalCase&#xff09;。 # 最简单的类定义 class MyClass:pass # 使用pass占位符类的定义就像是…

OSPF综合实验报告册

一、实验拓扑二、实验要求1、R4为ISP&#xff0c;其上只配置IP地址&#xff1b;R4与其他所直连设备间均使用公有IP&#xff1b; 2、R3-R5、R6、R7为MGRE环境&#xff0c;R3为中心站点&#xff1b; 3、整个OSPF环境IP基于172.16.0.0/16划分&#xff1b;除了R12有两个环回&#x…

网络层6——内部网关协议RIP、OSPF(重点)

目录 一、基本概念 1、理想的路由算法应具备的特点 2、分层次的路由选择协议 二、内部网关协议RIP 1、特点 2、路由交换信息 3、距离向量算法 4、坏消息传送慢问题 5、RIP报文格式 三、内部网关协议OSPF 1、特点 2、其他特点 3、自治系统区域划分 4、OSPF的5中分…