TA的每日心情 | 开心 2022-12-27 15:07 |
|---|
签到天数: 1 天 [LV.1]初来乍到
|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
每个C程序都有一个main()函数。当然,有可能编写一个成功的程序,其中唯一的功能是main()。我的猜测是已经做过很多次了,确实在某些简单的应用程序中不需要其他功能。7 L& b6 ^/ e" Q r% K' E
" w* [' i# `, i
但是,功能的广泛使用表明编写代码的人是经验丰富的固件开发人员。为什么?因为函数使我们能够更快地编写更好的代码,从而减少工作量并减少错误。对于那些花费大量时间编写固件的人来说,这些优势是不容忽视的。即使我们起初因为似乎需要更多工作而拒绝使用功能,但经验逐渐告诉我们,好处远大于成本。
' p! ^4 Y5 u$ h8 n) R. i( f$ ]/ ~4 F. k
什么是功能?. s5 R& |6 _/ \1 w' G
- J2 D3 L- V% L* [9 b* J
AC功能是一组指令,这些指令一起工作以实现特定类型的处理器活动。在许多情况下,某个功能会执行一项特定任务,例如从SPI缓冲区中检索数据,配置计时器以使其产生指定的延迟,或者从存储器中读取值并将其加载到DAC寄存器中。
3 r5 _ w2 A+ E( N" D但是,当然没有法律说明某个功能只能执行一项任务。您可能会发现拥有一个更新三个不相关状态机的函数很方便,或者您可以编写一个通过UART发送一个字节,然后检查状态位直到接收到一个字节,然后将接收到的字节的值合并到一个函数中的函数。数学计算。+ Q5 [5 O- J: g5 u1 |
$ [- [7 ]+ ?& a4 ~" D
1 ^- r+ \8 `' ?7 i* U我喜欢函数的一件事是,它们使流程图和代码之间可以进行相当直接的转换。
0 \# Z- s0 t! f功能的“组成部分”函数由名称,输入参数列表,实现所需功能的代码语句以及返回类型组成。
* T, ^. I# [, I) [. J以下代码段为您提供了一个示例。
+ ^+ G: L6 t- z7 E' D& [# e( `. M% X
) L h9 q8 q, {2 A" E5 ~! S
4 {+ Z0 a$ t6 z# H# z3 E) ?; y+ _9 u) |$ m
- char Convert_to_Lowercase(char UppercaseLetter)
- {
- if(UppercaseLetter < 65 || UppercaseLetter > 90)
- return 0x00;
- else
- return (UppercaseLetter + 32);
- }- u' F' G/ x3 D3 D! Z. x7 w, w
0 O/ P4 V6 U' T% l( C2 I2 b& W, J
0 V8 s# h% i) W& O& N1 C复制代码
( Q% O+ p7 S. c# h0 ^6 E) b3 j0 p* Q0 q& a
6 d' q3 K' T$ [+ w我喜欢使我的函数名称具有很强的描述性。这使代码更具可读性,并有助于您保持思想井井有条。
+ R) L5 a: J: v/ z
4 V3 i0 L* x. {/ i这些说明放在大括号中;函数定义的这一部分称为函数的主体。“ return”关键字用于退出函数并标识哪些数据应传递到代码的先前执行部分。7 ~7 T1 b" `; ]7 t3 \: S) I
返回类型位于函数名称之前,用于标识将要返回的信息的数据类型。具有仅执行任务而无需返回数据的功能是完全可以接受的。在这种情况下,您将使用关键字“ void”代替数据类型。! E/ }4 A5 B9 K- ^+ V: f
9 \. {9 W, ]/ K7 V1 W6 }
将数据传递给函数
9 w1 w2 s4 z9 H: d3 {$ H输入参数(也称为自变量)用括号括起来并放在函数名称之后。AC函数可以有多个参数,在这种情况下,它们用逗号分隔。每个参数必须带有数据类型。
; x3 |5 o( I# b在嵌入式应用程序中,通常不需要使用参数。我可以想到两个原因。& l3 v* @1 `5 Y2 p H( b
首先,嵌入式固件经常直接与设备的硬件交互,因此功能可以从配置寄存器,通信寄存器或端口引脚中获取所需的信息。
0 E' [8 K3 h# `' K其次,为微控制器编写的简单C程序可以使用全局变量,即整个程序中存在的变量,并且可以由任何函数访问。据我了解,不鼓励在应用程序编程中使用全局变量,或者甚至“定罪”将是合适的词。但是我认为许多固件项目,尤其是完全由一个程序员编写的固件项目,都可以从全局变量的简单性中受益。& }- j0 M0 Y+ S' ?" q# W6 X
定义不带参数的函数时,可以将括号留空或插入关键字“ void”。从理论上讲,“空”方法比空括号要好,但是在嵌入式开发的背景下(尤其是考虑到现代编译器的智能程度),我不知道它到底有多重要。
0 s: \2 r& x4 T* H; L, I K9 t, c" @4 l. j' _$ V5 Q
演练
$ d0 k: Z" H* c4 t* {+ b让我们简单地检查一下上面显示的函数定义。
5 L; M3 c1 q7 C6 v% R8 @- 函数名称Convert_to_Lowercase清楚地表明了该函数的用途:它接受与大写ASCII字母相对应的八位值,并返回与该字母的小写版本相对应的八位值。
- 有一个输入参数。它具有char数据类型,并使用描述性标识符。
- 与输入参数一样,返回值是ASCII字符,因此返回类型为char。
- 如果输入值超出了与大写ASCII字母相对应的范围,则该函数返回0x00,表示错误。否则,它将32与输入值相加并返回总和。如果您不熟悉ASCII值,下表将帮助您了解为什么我使用数字65、90和32。
+ K0 _4 j3 U, g+ L* ` 9 g* j" f2 v; f' S2 h. s
" W T$ N# ~6 Q0 K# C
. A& c0 `$ V! w' y3 o9 v# }
硬件功能, ]+ W1 b$ H4 @' o% L) |- ~0 M
正如处理器的数据存储器不直接支持附加到C 变量的详细信息一样,处理器的代码存储器比C函数要简单得多。代码存储器是很长的存储位置序列,没有以任何有用的方式对其进行分类或组织。唯一标识代码存储器中特定位置的是地址。5 z9 p Z% [' y$ A3 S8 A7 n
因此,AC函数是一种精心设计且对程序员友好的方法,用于将代码块放置在内存中并将处理器定向到需要执行的块。如果您使用汇编语言,那么您将熟悉代码执行的底层现实:每条指令都有一个地址。我们使用文本标签表示给定的地址,如果我们希望处理器在该地址执行指令,我们告诉它跳转到标签。! ^! g6 g4 L$ X$ w( K9 d
C函数是对汇编语言中使用的基本子例程的重大改进,但它们在本质上没有什么不同。当您调用或“调用”一个函数时,处理器的程序计数器将接收与该函数关联的第一条机器语言指令的地址。8 t( {3 ?& D4 z4 b
+ X( g/ t4 }3 T. [" L
调用堆栈
' L# Q# D2 D3 R; L3 ?内存的一部分称为调用堆栈,用于存储函数执行后处理器应返回到的代码存储地址。堆栈还为局部变量(即在调用函数时创建并仅在函数内使用的变量)提供了存储位置。
6 E7 d1 T' v& H5 ^+ @8 F) \4 H8 R6 H' s5 l; s
5 I" G5 [+ E8 K1 M调用堆栈的示例。请注意,该图没有指出每个项目需要多少字节。% O# N/ ^6 Y6 j7 }. x% G/ A* C2 S
调用堆栈可以用作存储作为输入参数传递给函数的数据的地方,但是据我了解,编译器将在可能的情况下为此目的而不是内存使用寄存器(因为寄存器更快)。
# b0 ^1 Q7 m5 A) e/ r3 T9 d/ X/ b- P6 [
结论* d0 C( \8 }/ b+ L: W! x
我希望本文能为您提供有关C语言程序和处理器实际硬件中函数的结构和行为的良好概述。在下一篇文章中,我们将继续讨论C函数。8 J0 b) Y2 @/ Y" `* v
5 D0 Q* ?% O3 u2 L3 E
|
|