在嵌入式开发中如何提高自己的代码水平$ U9 L0 w; a5 j, E
* e0 i" W+ u( a+ @, s
在开始学习嵌入式系统或者单片机系统的时候,大多数人的学习方法都是自己找视频教程或书本自己学,有的人也许会掏钱报个培训班什么的跟着老师学。% U% K4 v. |2 G7 W* I, o3 u1 n
6 t+ A- v. `2 Q4 p 当自己慢慢的可以独立写代码的时候,就会觉得已经学会了。特别是自己能独立完成一些简单的项目之后感觉自己可以出师了。慢慢的随着各种项目的锻炼,写代码的能力也会慢慢提高。写到最后就会发现,其实好多项目都是不同的代码重新组合,一个功能写好之后,在每个项目中都可以用。比如写好了一个串口功能之后,那个项目用到了串口,直接将以前写好的代码拷贝过去就完了,最后开发项目就变成了一个拼积木的过程。' |9 v) s8 y' z- n' ]
! H# s4 X& e" ~# O( x 这时候基本写代码好像就失去了最开始学习的激情,写代码就变成了一个体力活了,基本也不用动脑了,都是复制粘贴了。写代码的能力好像也很难在突破了,感觉到了瓶颈。这时候就很难找到视频或者书籍或者老师教自己如何突破这个瓶颈。此时如果想要自己的代码水平更进一步,要怎么做呢,下面我就将自己的经验分享一下。 / T6 q6 X7 J9 \3 V1 o4 ^ e0 n, O6 Q& h, |) y/ u# H0 Y
在自己写完一个功能之后,如果想知道自己写的怎么样,最好的办法就是找一个别人写的同样功能的代码去比较。看看自己写的和别人写的有什么区别。找代码的时候,不能网上随便找一个,因为你不知道你找到的代码是初学者写的还是高手写的。容易被误导,这时就要找大家比较认可的,代码风格比较好的去比较。我一般找到都是某原子和某火的代码。下面就通过一个简单的LED代码来分析比较一下。 ( W% V% m1 Z! V: {; q3 r3 Y+ }& \8 R3 j. w4 |4 |, Z
首先看某原子的代码 4 g! @* F9 f) ^5 b$ H4 T) v' ?/ ]! F( z9 ?6 U' X3 F$ F
这是 led.h 的代码 8 G0 v. r! ^% T8 O `4 y* J. x2 m' | P$ r
+ }0 A Q" v2 J7 X; r+ V G 这是 led.c 的代码7 n% D0 _' q9 ] k5 J' b* ~# [
8 |% r; q" `5 Z- a1 a O; X2 Y1 h6 Q+ k
led.h里面用到的位操作是在sys.h头文件中定义了。 0 l( |! C0 c( y" P4 ^ 3 U3 u; b9 x& y, E4 Y ! y) ^, d: Q) S5 B% e6 { 这个初始化代码看起来非常简洁,首先使能用到的IO口时钟,然后初始化相应的IO口。在初始化函数中一眼就能看出两个LED分别是PA2和PD8口。代码理解起来非常容易。& a. m: L5 f ]5 M* \
V3 f- U& T1 J7 i' d P 接下来看某火的代码 3 }: @1 y! H: U" \4 m 头文件 bsp_led.h * w$ V: J, C" P" |, L' W . C$ Q7 G% |# h6 _/ `$ C: q7 R+ R: v3 e2 |4 _6 h1 A
7 Y$ w# d* S0 n1 s, k C文件 bsp_led.c , ?+ s# S( _1 k: E& e 4 Y- E2 p( d- [: K! C1 x0 X( T5 E" h, L8 S |. M6 a
* G4 V; g0 \3 c' p7 Z$ `" e
首先从文件名字就能发现区别,上面的是led.h,下面的是bsp_led.h。前面加了一个bsp,这个bsp是什么意思呢?BSP即Board Support Package,板级支持包。也就是这个文件是和当前硬件系统有关的。然后在看头文件中的内容,发现在头文件中,把LED用到的IO口和具体引脚重新定义了一个名字。然后在C文件中初始化LED口的时候,用的都是新的名字。初始化的流程也是先使能用到的IO口的时钟,在初始化用到的相关IO口。但是从初始化代码中很难看出,当前使用的LED口是哪个端口。代码看起来感觉就比较费劲。' | n! L4 L9 V& \, T2 [3 P' s! {
9 u0 H! c; F$ [4 @! J) X 为什么一个简单的LED功能,风格却相差这么大呢?这两种风格你自己的代码更接近于哪一种呢?从我自身来说,刚开始的时候比较喜欢上面的那种风格,简单明了,代码写起来很容易。但是随着写代码的时间增加,逐渐开始喜欢下面的这种风格。代码也由上面的风格逐渐转换为下面这种风格了,下面就分别讨论一下这两种风格的差异。) W6 k2 p: B; H0 K
* G N5 t y$ B3 s5 G% J 刚开始写代码的时候,第一种风格比较适合,写起来比较简单,容易理解,代码出现问题时查找起来也更加方便快速。但是随着项目的增加和需求的各种变化,就会发现,第一种代码修改起来比较麻烦,比如LED口发生了变化,那么头文件和和初始化函数都需要修改,变动一次要修改好几个地方,修改的时候容易漏掉有些地方,导致调试起来各种问题不断,增加了调试的难度。这时就想,能不能值修改一个地方,就能把LED口用到的地方都修改过来。此时就需要用到第二种代码的方式了。在头文件中将LED用到的端口和时钟都抽象出来,重新定义一个名字。在初始化代码的时候,初始化的只是是LED口,至于这个LED具体对应到哪个口,直接在头文件中定义就行。这时候初始化代码就不需要修改了。 + N( Q7 y8 z- S$ r/ |( ~- W, s2 W- D% P6 \' L5 j7 q1 w& D
简单的理解就是,第一种方法写起来简单,针对的是具体的IO口,但是一旦项目发生了改动,修改和移植起来比较麻烦。第二种方法写起来复杂,但是是一个抽象的方法,它的初始化函数和具体的IO口没有关系。当项目发生变动或者移植起来更方便。0 p p- \6 B9 @
8 }$ V( f8 d4 @- U% m. y* ` 这两种方法各有优缺点,可以根据自己的实际情况选择不同的方法,也可以将这两种方法结合起来。比如我感觉第一种方法的位带操作比较简单,将所有的位操作统一定义在一个文件中,使用起来比较方便。第二种方法的位带操作都卸载头文件中,写起来比较麻烦,也比较复杂。但是第二种方法把LED口使用的IO口和引脚抽象出来比较好。那么就可以将第一种的位带操作和第二种的端口抽象同时用在自己的项目上。 8 Y, T1 ]# S. F8 C/ R! h# F7 Q0 _$ t0 J( I$ z3 L! V
这样通过对别人代码的分析,然后结合自己的代码,对比不同代码的优缺点,然后再修改自己的代码。不断的模仿别人优秀的地方,自己的代码也就写的越来越好了。如果感觉自己的技术比较好了,那么就可以去看库函数的源码,或者操作系统的源码,看看官方的源码是如何将不同类型不同型号的单片机抽象为统一的函数。如果哪天你看到别人的代码之后,一眼就能看出代码的优缺点时,那么就说明你的水平已经很高了,已经突破了自己的瓶颈了。) N, s# Z+ f, T( Y: k3 R
———————————————— 4 r6 Y& \* Q: X, U. U; _' E版权声明:本文为CSDN博主「嵌入式@hxydj」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 ) z9 ^$ F( `- K u0 c原文链接:https://blog.csdn.net/qq_20222919/article/details/126748702 3 O0 d2 Y* G1 I3 m9 P2 u. o2 T: d0 c& x/ [/ {: s9 H" m
+ T7 J/ r% N' `! [2 p9 ?9 Z% q