如何在嵌入式C语言编程的针尖上舞蹈?15年大神7个实例讲代码

单片机爱好者 2022-07-28 18:30

整合自网络信息,作者:小马儿


不记得从何时开始,自己就经常承担企业招聘的工作,侧重于工业嵌入式产品研发人员的招聘。


出于对企业和新人的负责态度,我倾向于寻觅那种基础知识扎实,且对嵌入式编程有兴趣的人员,这样对企业发展、对个人成长都有好处,可惜很多次都如同大海捞针,希望而来失望而归。


这是我很纠结的一个问题,不由得想起了曾经听过的一个故事:一个外国人来国内某公司考察,首先看到了职员的工作态度,狂吐槽,这样的员工企业应该全部裁掉;然后中午吃饭的时候,品尝了企业的午餐,又开始骂企业了。


回头看一看目前的大学教育模式,很多学校还是沿袭类似于初高中的填鸭式教育方式,大多数学生也是在吃喝玩乐混日子,即使大一新生还有些激情,很快就被大环境给污掉了。然后就是学生、学校和企业的互骂,搞得一片狼藉,戾气冲天。


一个要好的朋友在国内某985大学教计算机,一次和他聊天埋怨到,现在的学生入职后好长时间都没法上手,培训周期很长,然而朋友的观点是,这应该是理所当然的啊,大学能学点啥,都需要到企业才能锻炼的。无语……


真应该是这样吗?我很茫然,但看到每个学生背后为学费辛勤劳作的父母,看到他们对子女期望的眼神,我认为,我们可以做的更好,也应该做的更好。


为了让大家体会我内心纠结的心情,先简单的聊一聊招聘试题的事情。
以前我参加过中兴的招聘,招聘老师给我出了一道题目,int n = 3,4;我说我不知道,这是变态的用法,我们不应该这样用,您也不应该出这样的题目。可能当时年轻气盛,将招聘老师气了个半死,我当然也无缘中兴了。吸取以前的教训,后来我在组织招聘时,有一个基本原则,只有在工作中经常会用到的知识,才会作为考察的范围。


为了挖掘一些好的苗子,我们也一直在思考怎样的试题可以更好的考察出求职者的能力和兴趣。最开始的方式是先问一些基础课和专业基础课的基本概念,滤除基础知识不扎实的人;然后期望求职者能讲解一个自己曾经凭兴趣写过的程序,然后借着求职者的描述,然后不断的深入,以判断他的能力具体到了什么程度。


可惜,99%的人根本说不出自己凭兴趣写的程序,碰上几个胡诌的人,几句话就露馅了。后来只好雪藏该问题,然后降低难度,出一些单层循环,且每层循环适度判断的编程例子(如判断一个整数中0的个数等)。如果每次招聘,能碰上一两个将这类程序快速写出来的同学,大家都可以兴奋半天了。


这就是我们能招聘到的学生的普遍能力了,在我看来,要达到这个能力,只要随意的学习几天编程就可以了,还需要大学四年干嘛,让人郁闷啊。
但一个更加现实的问题是,这样的人招聘进来企业如何使用。我们不敢直接拿着产品让菜鸟人员练手,大家都知道,产品是要卖给用户的,初期的小问题到了用户那儿就会放大成大问题的。


企业培养不同于学校教育,不会有人给你填鸭式,更别指望有人手把手的教你,很多时候都是散养,顶多指点几句然后自己看书去。这种情况也导致了很多人初期很难上手,一些人甚至被迫转行,荒废了自己多年的专业方向。


如何快速过渡,结合自己多年的工作和新人培养经验,我总结了一套比较有效的方式,将产品研发需要的初级C语言知识巧妙的融入到三个例子中,让新人通过这三个例子去学习,去碰壁,去思考,持续的提高自己的C语言能力,尽快具备可参与产品研发的能力。


如何学习C语言?


很多刚入职的新人,都喜欢问一个类似的问题:“如何学习……?”,然后一些朋友就会给拷贝一大堆书籍资料,更热心的还会指导先看哪本在看哪本。

但不幸的事,很多时候就截至于此了,一大堆资料依然静静的躺在电脑硬盘里,只是在偶然的情况下才会打开翻看两眼目录。

走过一些企业,培训体系一般是这样的:


1、新人入职后,师傅会给一堆资料让看,然后新人硬着头皮看一些;
2、哪天师傅不忙了,惦记起这个新人,然后交给其一个产品,让其折腾;
3、可惜具体产品一般都涉及多个学科,面对一大堆疑问,新人会感觉腾云驾雾般难以前行;4.一段时间后部分人迈过了入职时的绝望悬崖,有了自己的积累,开始慢慢的深入接触产品,但因各种文档资料奇缺,只能一边学习一边调整;
5、数年后,新人成为了老手,同时新的产品体系也诞生了;
6、然后重复以上死循环。


长此以往,公司的产品体系变得非常的杂乱,技术难以复用,无法进行有效的积累,那种如臂使指的团队构建更是空谈。

如何才能跳出上述死循环呢?我在自己的职业生涯中进行了大量的探索尝试,有了一点感悟:不仅需要将新人的培训工作尽可能前移,尽可能体系化,而且要尽早的融入我们的设计及团队理念。因此在后续的三个C语言例子中大家会体会到很多东西不仅仅是C语法层次的内容。

前面我谈到,在公司,不可能在重复学校的填鸭式的教育模式,也不会有人手把手的教你。那么在这种情况下,如何组织培训工作,很多时候反而成了一件颇具艺术感的选择难题。


随着移动互联网的盛行,在线教育也流行了起来,网络上出现了大量的视频教程。紧随时代的脉络,我尝试过将入职需要学习的内容做成书籍和视频,但效果一般。后来细细研究这个领域,发现大多数视频仅仅是将大学课堂的内容移到了网络上,很多人也仅仅是耐着性子看个几集,然后就不了了之了,因此导致的效果不佳。关于在线教育我也有一些自己的想法和思考,后面会专门撰文描述,这儿就不深入介绍了,总之,初期的尝试是不太成功的。


C语言第一个例子:需求不明确


第一个例子:“编写一个控制台程序,已知内层和外层菱形的高度,输出一个空心菱形”。

是否看上去很简单啊,写到这儿,很是期望正在阅读的你能先停下来,思考思考,然后付诸行动,编写几行代码小试一番,然后在继续读下去,收获会更大。

这些年,我带过很多人,大部分人看到这个题目后,然后立即就开始写程序去了,自己内心多少有点小小的失望,为何?


一般通过公司层层把关招聘进来的软件人员,这个例子总是可以弄出来的,不过可惜的是,近一半以上的人不能准确的实现出来,仅一个菱形高度的理解,就是五花八门,乱七八糟的,执行如这样的:


或者这样的:


或者干脆是这样的:


为何会出现这种情况,这恰恰是该例子的第一个大坑:需求不明确。一开始就碰到挫折,对很多新人来说,这无异于当头一棒,不过记忆也最为深刻。

一个团队协作时,会存在大量的交流,而交流过程中,总是会产生或多或少的歧义,而恰恰是这些歧义会导致需求的不明确,会导致大量的返工,甚至会导致项目的失败。我给新人的第一份建议:将需求用自己的语言表达出来,和对方确认后再实施。这一点在以后的团队协作中非常的重要,因此我早早的将这一点嵌入到了入职培训中。

经过这么一折腾,第一个例子重新描述如下:“写一个控制台程序,用户输入内层和外层菱形的高度,输出一个空心菱形,菱形的高度定义为菱形的上三角形的高度,如输入5和3,输出如下:
  
      *

    ***
  **  **
**      **
**      **
 **   **
  ** **
   ***
    *


这儿稍微补充两点:


1、菱形高度定义好似有违常理,但在编程的世界中,简洁就是常理,如果定义为整个菱形高度,那么不仅需要进行正整数判断,而且还需要奇数判断,增加了程序的复杂度。

2.、用一个例子描述,比一大堆文字管用多了,是所谓一图顶千言。

明确知道要做什么了,然后很多人又回去,然而很多情况下执行结果如下所示:


这儿引出了第二个很重要的概念:边界判断。给出两个数,如何简洁且完备的判断其是合理的输入,这是编程的基本功,这些内容在大学教学过程中一般不太重视,但想做出合格的嵌入式产品,这一点必须引起足够的重视。如何做出优雅的边界判断,这个例子比较简单,留给正在阅读的你吧,顺便回头想一想我前面菱形高度的定义吧。

第二次被打道回府后,很多人会有急躁情绪,菱形还没输出呢,就给灌输了一堆规则。碰到这种情况,我会给他们讲解嵌入式程序的特点(这一点留待后文慢慢描述),以及我以前的经历。我刚开始工作时,因CPU速度受限,程序还主要是汇编语言,领导让写一小段汇编程序,但每次提交后,都被劈头盖脸的骂一通,让回去整改,直到后来一行汇编语句都省不下去了,才算通过了。回头看,为了一个小小的例子,竟然将大多数汇编指令熟悉了,方才明白领导的良苦用心。现在的新人脸皮薄了一些,不敢乱骂了,只好慢慢讲道理了。

经过第二次的折腾,大部分新人态度都能稍微端正起来,而且这次写出来的例子,和需求基本吻合了,但当兴致冲冲的跑过来交差时,我反而不看程序,开始问起在实现这个程序过程中用到的调试技巧。

这个例子对新人来说有一点点的难度,不可能一次就写好的,肯定会经历一些痛苦的调试过程。但是在国内的大学教育体系下,却又不太重视基本调试技能的锻炼,期望新人一开始意识到调试是一项基本功夫,需要引起足够的重视,需要去持续的加强。

雄关漫道真如铁,而今迈步从头越,一个小小的例子还未起步,对很多人已是一场痛苦的经历。要想成为一个合格的嵌入式工程师,需要方法,需要才智,更需要背后的汗水和坚持,本系列文章会带领大家体味我的成长路,但无法代替你自己的付出和汗水。


C语言第二个例子:边界判断


上文中,我提到由该例子引出了三个重要概念,其中第二个概念是:边界判断。给出两个数,如何完备且简洁的判断其是合理的输入,这是编程的基本功。很可惜的是,这几个例子都没有做到。结合自己多年带人的经验,这儿补充阐述一下。

输入两个数,一个是外菱形的高度(m表示),一个是内菱形的高度(n表示),有如下几个判据:


1、两个高度都应该是正整数(内菱形高度可以为0);
2、受限于屏幕的大小,菱形高度应该受限;
3、外菱形高度应该大于内菱形的高度。


以前带人的时候,拿到例程后,我喜欢先输入(1000,800)这样的值,因为这是容易被忽略的地方,很多人郁闷的铩羽而归。

大部分人都是缺判据,也有一些人性格比较谨慎,习惯写一大堆的判断条件,如下面这种的:


if (m > 0 && n > 0 && m > n && m < 30 && n < 30) {   ……}

总之,都没有抓住判断条件应该完备且简洁的基本准则,实际上该例子很简单,只要简单的分析,判断条件也就三个,如下:


1、内菱形高度大于等于0;
2、外菱形高度小于约定之(假设30);
3、外菱形高度大于内菱形高度;


写成程序示意如下:


if (n >= 0 && m > n && m < 30) {   ……}


还记得第(1)节中我们约定菱形的高度是上三角形高度吗,带来的好处就是判断的简洁化,概念是为目标而服务的,不然该处的判断还需要额外的增加两条奇数判据了,简洁性也会打折扣了,呵呵,可以再回味一番。

在嵌入式产品中,最终产品的鲁棒性,很多时候就是表现在这样一点一滴的简单判据上,该处的不厌其烦,也是期望新人慢慢的融入研发团队时,能够充分意识到这一点。


终于要开始输出空心菱形了,但对于刚毕业的大学生,这个例子刚上手还是有一些绕的,其思维逻辑是如何的呢。


一般人都会发现,输出空心菱形要稍微复杂一些,那么我们修改为输出菱形呢,是否简单很多,如果还嫌复杂,修改为输出三角形呢。呵呵,说白了,就是将复杂的问题去其枝叶,先简后繁的慢慢处理。


一开始的问题就简单多了,已知三角形高度m,输出三角形,如m=5,输出如下:


    *

   ***

  *****

 *******

*********


为了直观,将空格也表示出来,示例如下:


----*

---***

--*****

-*******

*********


此时,结论已经很形象了,每行输出的空格从m-1递减,每行输出的*从1开始递增,循环子为m,程序示例如下:


for (i = 1; i <= m; i++){  j = m - i;  while (j--)    printf(" ");

j = i * 2 - 1; while (j--) printf("*");

printf("\n");}


问题复杂化,考虑空心三角形,相当于里面又多了一个三角形,我们输出的时候,将其简单扣去即可,假设n=3,如下图示例:


----*

---***

--**-**

-**---**

**-----**


程序示例如下:


for (i = 1; i <= m; i++){  j = m - i;  while (j--)    printf(" ");  j = i * 2 - 1;  for (k = 0; k < j; k++)  {    if (k < m-n || k >= j-m+n)      printf("*");    else      printf(" ");   }   printf("\n");}


进一步复杂化,输出完整的空心菱形,仅仅需要将上面的程序重复一下,仅仅是将其颠倒一下,且高度调整一下(减1处理,表达式更加复杂了)而已,我就不展示示例程序了,大家有兴趣的可以自己尝试一下。


有些人在实现该程序的时候,一开始就是对空心菱形进行分析,最直观的分析策略就是将菱形分层了三段,上三角形,下三角形,中间的空心部分,如下图示意:


    *

   ***

  ** **

 **   **

**     **


 **   **

  ** **

   ***

    *


按照这种思路,程序示意如下:


for(i=1;i<2*m;i++){    if(i<=m-n) {        star = 2*i-1;        empty= m-i;        while(empty--)            printf(" ");        while(star--)            printf("*");    }    else if(m-n        if(j <= n && i <= m){            num_empty = 2*j-1;            empty = m -i;        }        else{            num_empty = 2*(2*n-1-(j-1))-1;            empty = i-m;        }        num_star = star = m-n;

while(empty--) printf(" "); while(star--) printf("*"); while(num_empty--) printf(" "); while(num_star--) printf("*"); j++; } else { star = 2*(2*m-1-(i-1))-1; empty = (2*m-1-star)/2; while(empty--) printf(" "); while(star--) printf("*"); } printf("\n");}


不管如何,至此,这个例程就算完成了,但大家有没有发现上面这些程序都谈不上优雅啊,其中各种m和n的表达式,一段时间以后看,基本同乱麻差不多了,试想,如果这是产品的程序,让后来人如何阅读并维护。


C语言第三个例子:简单粗暴的数字之美


优美,总是让人心醉,一提到优美,最容易想到的是悦目的图画,动听的乐章、精妙的诗文……。然而,数学,自然科学的皇后,却蕴含着比诗画更多的优美。

优雅的程序,或许其背后都蕴藏着数学的优美。

在上一节中描述的一些例子中,我们总是在努力的拼凑各种m和n的表达式,与其这样苦苦寻找,为何不直接将这个空心菱形放入坐标轴中呢。

在电脑屏幕上,人们习惯将靠右称之为x轴,靠下称之为y轴,将空心菱形画在屏幕上,示意如下:

然后通过解析几何知识勾勒空心菱形,程序示意如下:


for (x = 0; x < m * 2 - 1; x++){    for (y = 0; y < m * 2 - 1; y++)    {        if (abs(m - 1 - y) <= m - 1 - abs(m - 1 - x) &&            abs(m - 1 - y) > n - 1 - abs(m - 1 - x))            printf("*");        else            printf(" ");    }    printf("\n");}


我们将所有的判断都集中在了一起,阅读程序,很容易明白这个大大的判断语句是干嘛的了,是否比以前的实现都优雅了很多呢。


不过这个判据好像还是挺复杂的,有没有更好的办法呢,估计很多朋友在看到我上面的那幅图时已经想到了,那就是将坐标轴移到菱形的中间去,示意如下:

外菱形的四条边我们用表达式描述出来,如下:


(+x) + (+y) < m(-x) + (+y) < m(+x) + (-y) < m(-x) + (-y) < m


合并后的表达式为:abs(x)+abs(y)


for (x = -m; x <= m; x++){    for (y = -m; y <= m; y++)    {        t = abs(x) + abs(y);        if (t >= n && t <= m)            printf("*");        else            printf(" ");    }    printf("\n");}


不知大家看到这段代码是怎样的感觉,我仅记得当初自己发现这个实现后,第一次被这种简单的数学美给震撼了。如果大家也有相同的感觉,我坚信,你可以在编程的这条荆棘路上走很高很远…


C语言第五个例子:嵌入式C语言和桌面C语言差异巨大


在大学阶段,我学的是计算机专业,对编程的爱好聚焦在各种巧妙的算法和实现上。工作后,开始从事嵌入式产品开发,碰了N多的壁,吃了N多的苦,才明白了嵌入式C语言和桌面C语言是有差别的。

为了让新人尽快迈入嵌入式门槛,因此,今天,注定是一场批斗大会,经历波折,方能成长。

我们首先拿最优秀的实现开刀,在上一节最后,分享了一个很优美的实现,示例如下:


for (x = -m; x <= m; x++){    for (y = -m; y <= m; y++)    {        if (abs(x) + abs(y) >= n && abs(x) + abs(y) <= m)            printf("*");        else            printf(" ");    }    printf("\n");}


该实现将最复杂的条件判断置于两层循环之中,因此程序效率很低,而在嵌入式编程中,尤其是一些强实时模块,性能经常是紧绷着的弦。

工业嵌入式产品研发,追求团队作战,追求产品可维护性,追求性能和资源的均衡。可读性、可维护性、cpu计算能力、内存、可靠性等等东东都成为了资源,需要我们的庖丁解牛,经常,我们喜欢将工业嵌入式产品编程戏称为针尖上的舞蹈。

除了这一点,前面的程序还有很多的不足,举例如下:


1、整个程序经常混杂一谈,增加了他人阅读程序的复杂性;
2、m,n类似的名字,典型的学校风格;
3、各种复杂的m和n的表达式,一段时候后,估计自己看起来都头大了;
……


因此,我们需要继续上路,为了后续程序实现的方便,将整个程序结构约定如下:


int main(){    /* 输入内外菱形高度,并进行合法判断 */
/* 循环输出菱形 */ for (……) { /* 输出前导空格 */
/* 输出左边星号 */
/* 输出中间空格 */
/* 输出右边星号 */
/* 输出换行 */ printf("\n"); } return 0;}


看到这个程序框架,容易明白一点,我们要求以单层循环的方式输出菱形。

细细观察空心菱形的结构,每一行从左到右可以抽象为前导空格,左边星号,中间空格和右边星号四部分,只需要找出这四个值和行号的函数关系,整个程序也就迎刃而解了,如下图示意:


----*
---***
--**-**
-**---**
**-----**
-**---**
--**-**
---***
----*

我的职业导师经常会和我探讨计算机编程思维的概念,时至今日,我们也没有办法给其下一个准确的定义,但在反复的迭代锻炼中,慢慢的体会到了哪些实现可以称之为计算机编程思维。

举一个例子,在计算机世界中,我们经常仅鼠标、键盘、磁盘、磁盘上的数据等等都称之为文件,可以进行简单的遍历。为何风牛马不相及的东西,我们非要将其抽象成统一的概念呢,这非一句话能描述清楚,后续会带着大家慢慢体悟,但这种设计思想在产品研发过程中,却比比皆是。

插入这两段废话后,回头再来看我们的菱形输出例子,明明每一行的结构不尽相同(回忆一下第二个实现版本,就是分类型分别输出的),但我们非要求同存异,或许,我期望从一开始,就在新人的心中播下架构设计的种子,期待着后续的萌芽。

虽然这儿的话题是以新人培训切入的,但未尝不是自己的成长之路,所不同的是我是在一次次的跌打滚爬中成长起来的。将自己曾经摔过的跟头融入到入职C语言训练的例子中,甚至将自己的整个成长史写出来,但愿别人能以我为镜,成长的更踏实更快速一些。


C语言第六个例子:嵌入式C语言和桌面C语言差异巨大


在上一节中,我们已约定了程序基本框架,并且也简单的介绍了编程思维的理念,好似,新人很快就可以写出合乎要求的程序了。

可惜,每个人的成长之路都不是一帆风顺的, 记得当初自己学习计算机编程时,哪个技能不是从大量的模仿中才能体悟一点点的,进而融入到自己的知识体系中的,一点一滴的成长起来的。

依据上一节的整体框架要求,我曾经带过的很多新人,甚至包含一些粉丝给我的发来的邮件,写出来的程序基本上都是新瓶装旧酒,仅是以前实现例子的翻版而已。

我给大家示意一下,大家体会体会,自己是否也有这样实现的冲动。

/* 循环输出菱形 */for (……){    /* 输出前导空格 */    if (上半菱形)       以三角型方式输出;    else       以倒三角形方式输出;
/* 输出左边星号 */ if (上三角形) ... else if (中心空心) ... else ...
/* 输出中间空格 */ if (上半空心) 以三角型方式输出; else if (下半空心) 以倒三角形方式输出;
/* 输出右边星号 */ if (上三角形) ... else if (中心空心) ... else ...
/* 输出换行 */ printf("\n"); }


注:该处的示例是结合第二个实现版本的,将一个菱形分为上三角形,下三角形,中间的空心三部分分别输出,详情请查看《入职C语言例子一(2)》。

应该如何实现呢?在上一节中,我们提到了重点在于寻找四个函数关系。文字的表现力经常是苍白的,给大家展现一幅我指导新人时的随意手绘图,或许,大家立即就明白了。


理解了上图,额外强调几点:


1、空心菱形的每一行都必须等同看待,忘记上三角、下三角、中间空心等这样的分割;
2、四个函数关系内部不允许出现if语句,但允许提炼公共函数,如abs类的;
3、成功没有捷径,技能的学习也没有捷径,将浮躁的心放下来;

走到这儿,很多人都写了七八遍了,拜目前的大学教育模式所赐,很多人又出现了浮躁情绪,即时的安抚还是很有必要的,经常打趣的一句话就是瞧瞧你的师兄xxx,别看现在负责好几款产品得心应手,当初还写了十多遍呢,哈哈。

实际上很多人也意识到了我这样折腾的目的。大学的学习都是浅尝辄止的,很多东西都急于求成,但将这种心态带入企业,带入产品研发中却是致命的,我仅是想通过这样的形式,让新人少走弯路。用心良苦,却常引来误解无数,内向者口是心非肚子里骂几句,张狂者会直接诘问我这样折腾他们有何意义,呵呵。

随意牢骚了几句人生坎坷路啊,经这样一指点,大部分人都可以按照要求顺利的完成该例子,虽然每一部分都是一个复杂的表达式。

此时,我会给大家讲解一个很关键的计算概念:迭代。

我们所从事的嵌入式产品是强实时工业产品,不仅经常涉及傅里叶等各种复杂计算,而且还要求在指定时间内完成,因此对计算效率等要求会比较高,最经常采取的策略就是通过迭代,保留有价值的中间计算结果,减少整体计算量。

而通过迭代,不仅各部分的表达式不至于那么的复杂,而且会减少计算量。该例子比较简单,没什么技术含量,大部分人很快的就完成了迭代的调整,但前后对比的效果却印象深刻,算是额外的收获吧。

至此,空心菱形程序的所有技术点都描述完毕了,按着这样的要求,大多数人可以写出满足要求的程序了。还是非常的建议正在阅读的你能亲自写一写,调一调,下一节我会贴出标准答案,大家可以在比对一下,而差异部分正好是我们后续内容的起点。


C语言第七个例子:不断完善心中的答案


前面几节中,列举了一个很优雅的实现,一些朋友提醒我该程序输出不正常,自己测试了一下,确实如此。

当时写程序时,是直接以文档的方式写的,一些例程也是小伙写的程序中拷贝出来的,重在意图表现,所有的程序代码都没有测试过,缺乏了严谨性,优化如下:


for (x = -m+1; x < m; x++){    for (y = -m+1; y < m; y++)    {        t = abs(x) + abs(y);        if (t >= n && t < m)            printf("*");        else            printf(" ");    }    printf("\n");}


咱们书接上节,言归正传,在上一节中,新人磕磕碰碰的,终于写出了符合技术要求的程序,皆大欢喜,以为要完工了,可惜,路依旧漫漫。

此时,我会给大家分享该题目的标准答案,让大家同自己写的程序进行比对,以前的程序都是以片段方式提供的,标准答案以完整的格式提供,示意如下:

/********************************************************************* *   Copyright (C), 1999-2004, xxxxxx. Co., Ltd.* *   文件名称:diamond.c*   软件模块:空心菱形输出*   版 本 号:1.0*   生成日期:2003/3/23*   作    者:xiaomaer*   功    能:空心菱形输出,该程序占用内存小,但计算稍大,可修改为"内存换资源"算法* *********************************************************************/
#include
/* 菱形最大高度 */#define MAX_DIAMOND_HEIGHT 16
/* 提前申明 */int myGreater(int n);
/* 主程序 */int main(){ int n, nRow; int nIn, nOut; int nCount1, nCount2, nCount3;
/* 输入内外菱形高度,并进行合法判断 */ for (;;) { printf("输入内外菱形高度(最大%d行):外菱形高度,内菱形高度:\n", MAX_DIAMOND_HEIGHT - 1); scanf("%d,%d", &nOut, &nIn); if (nIn < nOut && nOut < MAX_DIAMOND_HEIGHT && nIn >= 0) break; printf("输入不合法,请重新输入:\n\n"); }
/* 循环输出菱形 */ nIn = nOut - nIn; /* 调整为菱形内外差值 */ for (nRow = -nOut+1; nRow < nOut; nRow++) { //行号 printf("%-010d", nRow);
/* 输出前导空格 */ nCount1 = nRow >= 0 ? nRow : -nRow; /* 取绝对值 */ for (n = 0; n < nCount1; n++) printf(" ");
/* 输出左边星号 */ nCount1 = nOut - nCount1; /* 外三角部分,后续迭代使用 */ nCount2 = myGreater(nCount1 - nIn); /* 内三角部分,后续迭代使用 */ nCount3 = nCount1 - nCount2; /* 内外之差为实际需要输出 */ for (n = 0; n < nCount3; n++) printf("*");
/* 输出中间空格 */ nCount3 = 2 * nCount2 - 1; /* 由三角形拓展为菱形 */ for (n = 0; n < nCount3; n++) printf(" ");
/* 输出右边星号 */ nCount1--; /* 外三角部分 */ nCount2 = myGreater(nCount1 - nIn); /* 内三角部分 */ nCount3 = nCount1 - nCount2; /* 内外之差为实际需要输出 */ for (n = 0; n < nCount3; n++) printf("*");
/* 输出换行 */ printf("\n"); } return 0;}
/* 取大于0的数 */int myGreater(int n){ if (n < 0) return 0; return n;}


不知正在阅读的你看到这个标准答案的感觉,能否寻找出和自己程序的差异的地方,下一节我们以此程序为起点,给大家介绍一些真实产品中的代码特点。


撇开大道理讲嵌入式代码


在上一节中,我们给出了标准答案,是以真实产品代码风格写的,期望小伙伴们能同自己的实现比较一下。

现在的90后是有个性的一代,很多人都非常反感直接的大道理灌输,鉴于此,我期望我们的小伙伴们能自己需寻找答案,然后大家在交流碰撞中成长,效果或许会更好一些。

这篇文章就让我们一起来找出标准答案中有价值的地方吧。

1. 有意义的变量命名

大学老师教编程的时候,重点精力都放在了语法方面,侧重于将所有的语法给大家展现一下(这种学习方法我相当不赞同,后续会展现自己的方法,项目组内俗称大树法则的方法),因此经常使用短小的程序展示语法,但因为程序短小,因此变量命名也就随意了一些,因此i,j,k,m,n就成了常客,然后不小心带入了产品中,然后……。

但在产品研发的时候,即使比较简单的设备,代码量也会比较大,为了代码阅读维护方便,有意义的名字就变得非常的重要了。在标准例子中,使用了nIn和nOut就是想用简单的英语单词表示内外菱形的高度。

一些朋友可能读过《可读代码的艺术》或《代码整洁之道》等书籍,作者强调使用准确的英文单词来表达特定含义。

但我们是中国人,能想起一些简单的单词词汇已经颇为不易,想准确表达更是天方夜谭,因此,项目组内经过了无数次的迭代和探索后,形成了一种变量定义习惯:尽可能使用简单的相近英文词汇,全局变量必须加准确含义的中文注释,函数内的一些自动变量,因其作用范围很小,有时候注释可省略。

关于变量命名的故事还有好多,这个刚刚是给大家起个头,我们后面会有专门的系列文章介绍,记住在真实产品的代码中,需要有意义的变量命名,忘记m和n吧。

2. 代码分节

是什么是节(section),第一次知道这个概念的时候,正式全球跨千年的时候,我还在大四,我在北京一家企业打零工,当时公司承接了一个日本银行的项目,日方对代码质量要求很严格,专门派了一个专家过来给我们讲解各种要求,以及其背后的道理。

当时还很年轻,狂傲不羁,因此大部分的苦口婆心都被当做了耳旁风,但唯独对节的概念记忆比较深刻(可能是一开始就讲的是这个了,呵呵,后续的就没耐心听了,这个系列文章阅读比例逐次下降,估计是同样的道理)。

我们在读代码的时候,人的思维一段时间内仅停留在一个较窄范围的点上,如果面对的是看不到尾的代码,会潜移默化的将其看做灭绝师太的裹脚布——又臭又长,逆反情绪悠然而生。

因此,我们需要将代码按逻辑分成一块一块的,以空格作为区分,然后每块代码前增加适当的注释,解释这一块代码的功能,是所谓节的概念。

经过这样的改造,读代码的时候,感觉会完全不一样。关于节的价值,远不值这些,后续会在编程规范系列文章中和大家慢慢道来,此时,我们仅要求小伙伴知道垒又臭又长的代码是不对的。

实际上在第5节中,我约定程序整体结构时,已经有这样的意图了,大家不放回忆并体味一下。


/* 循环输出菱形 */for (……){    /* 输出前导空格 */
/* 输出左边星号 */
/* 输出中间空格 */
/* 输出右边星号 */
/* 输出换行 */ }


3. 细节化标注

某些代码存在着一定的深度,一段时间就会忘记,通过右侧简单的注释加以标注,不仅便于后续代码的阅读理解,而且标注点一般是关键代码段,给后续的代码审查也带来的方便。

在示例代码中,会看到迭代表达式右侧(微信公众号排版问题,经常到了下面一行)有简单的标注,主要就是起这样的作用的。

但万事过犹不及,很多刚入职的小伙伴喜欢在右侧加好多的注释,仅挑出有价值的进行标注,需要长期的锻炼和慢慢的体悟,或许,那一天回头审视自己的代码,会将许多无用的注释删除的时候,就修炼到家了。

4. 资源

嵌入式系统中,资源是一个需要时时刻刻关注的问题。

何为资源,在我们的概念中,不仅内存和flash空间大小是资源,cpu计算能力也是资源,甚至代码可读性(可维护性),代码实现复杂度(耗去的人力成本),复用率等等诸多方面,都被我们称之为资源。

好钢要用在刀刃上,但首先要明白刀刃在哪儿。缺乏了明确边界,空谈提高资源利用率是无意义的。如可读性第一位,内存和cpu资源比较宽裕,我们那个最优雅实现版本最佳了。如果cpu计算能力紧张,上一节的标准实现更好一些。如果内存稍微宽裕,为了增加代码可读性,我们还有更好的方法。

前面已经有人给我留言提到过这种方法了,不知大家有没有感受到,锻炼到现在这个时候,单纯的菱形输出是多么easy的事情啊,非要搞个空心菱形,将程序搞的混乱不堪。

但加入我们用一个数组来表示整个菱形输出,第一次以*输出一个菱形,第二次以空格在输出一个菱形,然后将整个数组输出出来,是否会非常的简单呢,代码可读性瞬间爆棚,执行效率也高,仅仅多占了一些内存而已。

总结


经过马拉松的历程,终于到了最后的篇章,我们来归纳汇总一下第一个空心菱形输出例程中提到的知识点:


1、需求清晰理解,最使用的策略是:将需求用自己的语言表达出来,和对方确认后再实施。
2、边界判断,要让小伙伴意识到:在嵌入式产品中,最终产品的鲁棒性,很多时候就是表现在这样一点一滴的简单判据上。
3、基本调试手段的锻炼,工欲善其事必先利其器,无须多言。
4、编程思维的引入,需要慢慢的体会抽象的价值,这是一个难点,但想走得远必须扛过去。
5、数值计算过程中,很重要的一个概念:巧用迭代。
6、基础编程规范的引入,体会节的概念,要意识到产品代码可读可维护的重要性。
7、在嵌入式编程领域,资源是受限的,而我们要学会针尖上的舞蹈。
8、撰写工作笔记,善于总结,习惯去体悟成长的脚步。


记得刚开始从事嵌入式编程的时候,我的职业导师给我欣赏了他的记事本,密密麻麻的各种调试记录,感悟想法,技术资料,知识归纳,我终于明白了他为何获得了全公司上上下下的认可。

因此,我们的团队形成了一条不成文的规矩,必须做工作笔记,不管方式,不管格式,只要开始记录就好。

单片机爱好者 《51单片机逆向学习实战教程》一书作者,单片机爱好者,每天为单片机用户分享一篇精品电子电路方面的经验笔记或教程。
评论
  • 每日可见的315MHz和433MHz遥控模块,你能分清楚吗?众所周知,一套遥控设备主要由发射部分和接收部分组成,发射器可以将控制者的控制按键经过编码,调制到射频信号上面,然后经天线发射出无线信号。而接收器是将天线接收到的无线信号进行解码,从而得到与控制按键相对应的信号,然后再去控制相应的设备工作。当前,常见的遥控设备主要分为红外遥控与无线电遥控两大类,其主要区别为所采用的载波频率及其应用场景不一致。红外遥控设备所采用的射频信号频率一般为38kHz,通常应用在电视、投影仪等设备中;而无线电遥控设备
    华普微HOPERF 2025-01-06 15:29 42浏览
  • 这篇内容主要讨论三个基本问题,硅电容是什么,为什么要使用硅电容,如何正确使用硅电容?1.  硅电容是什么首先我们需要了解电容是什么?物理学上电容的概念指的是给定电位差下自由电荷的储藏量,记为C,单位是F,指的是容纳电荷的能力,C=εS/d=ε0εrS/4πkd(真空)=Q/U。百度百科上电容器的概念指的是两个相互靠近的导体,中间夹一层不导电的绝缘介质。通过观察电容本身的定义公式中可以看到,在各个变量中比较能够改变的就是εr,S和d,也就是介质的介电常数,金属板有效相对面积以及距离。当前
    知白 2025-01-06 12:04 76浏览
  • 彼得·德鲁克被誉为“现代管理学之父”,他的管理思想影响了无数企业和管理者。然而,关于他的书籍分类,一种流行的说法令人感到困惑:德鲁克一生写了39本书,其中15本是关于管理的,而其中“专门写工商企业或为企业管理者写的”只有两本——《为成果而管理》和《创新与企业家精神》。这样的表述广为流传,但深入探讨后却发现并不完全准确。让我们一起重新审视这一说法,解析其中的矛盾与根源,进而重新认识德鲁克的管理思想及其著作的真正价值。从《创新与企业家精神》看德鲁克的视角《创新与企业家精神》通常被认为是一本专为企业管
    优思学院 2025-01-06 12:03 47浏览
  • 自动化已成为现代制造业的基石,而驱动隔离器作为关键组件,在提升效率、精度和可靠性方面起到了不可或缺的作用。随着工业技术不断革新,驱动隔离器正助力自动化生产设备适应新兴趋势,并推动行业未来的发展。本文将探讨自动化的核心趋势及驱动隔离器在其中的重要角色。自动化领域的新兴趋势智能工厂的崛起智能工厂已成为自动化生产的新标杆。通过结合物联网(IoT)、人工智能(AI)和机器学习(ML),智能工厂实现了实时监控和动态决策。驱动隔离器在其中至关重要,它确保了传感器、执行器和控制单元之间的信号完整性,同时提供高
    腾恩科技-彭工 2025-01-03 16:28 166浏览
  •     为控制片内设备并且查询其工作状态,MCU内部总是有一组特殊功能寄存器(SFR,Special Function Register)。    使用Eclipse环境调试MCU程序时,可以利用 Peripheral Registers Viewer来查看SFR。这个小工具是怎样知道某个型号的MCU有怎样的寄存器定义呢?它使用一种描述性的文本文件——SVD文件。这个文件存储在下面红色字体的路径下。    例:南京沁恒  &n
    电子知识打边炉 2025-01-04 20:04 49浏览
  • 在智能家居领域中,Wi-Fi、蓝牙、Zigbee、Thread与Z-Wave等无线通信协议是构建短距物联局域网的关键手段,它们常在实际应用中交叉运用,以满足智能家居生态系统多样化的功能需求。然而,这些协议之间并未遵循统一的互通标准,缺乏直接的互操作性,在进行组网时需要引入额外的网关作为“翻译桥梁”,极大地增加了系统的复杂性。 同时,Apple HomeKit、SamSung SmartThings、Amazon Alexa、Google Home等主流智能家居平台为了提升市占率与消费者
    华普微HOPERF 2025-01-06 17:23 22浏览
  • 根据Global Info Research项目团队最新调研,预计2030年全球封闭式电机产值达到1425百万美元,2024-2030年期间年复合增长率CAGR为3.4%。 封闭式电机是一种电动机,其外壳设计为密闭结构,通常用于要求较高的防护等级的应用场合。封闭式电机可以有效防止外部灰尘、水分和其他污染物进入内部,从而保护电机的内部组件,延长其使用寿命。 环洋市场咨询机构出版的调研分析报告【全球封闭式电机行业总体规模、主要厂商及IPO上市调研报告,2025-2031】研究全球封闭式电机总体规
    GIRtina 2025-01-06 11:10 62浏览
  • 本文介绍Linux系统更换开机logo方法教程,通用RK3566、RK3568、RK3588、RK3576等开发板,触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。制作图片开机logo图片制作注意事项(1)图片必须为bmp格式;(2)图片大小不能大于4MB;(3)BMP位深最大是32,建议设置为8;(4)图片名称为logo.bmp和logo_kernel.bmp;开机
    Industio_触觉智能 2025-01-06 10:43 60浏览
  • PLC组态方式主要有三种,每种都有其独特的特点和适用场景。下面来简单说说: 1. 硬件组态   定义:硬件组态指的是选择适合的PLC型号、I/O模块、通信模块等硬件组件,并按照实际需求进行连接和配置。    灵活性:这种方式允许用户根据项目需求自由搭配硬件组件,具有较高的灵活性。    成本:可能需要额外的硬件购买成本,适用于对系统性能和扩展性有较高要求的场合。 2. 软件组态   定义:软件组态主要是通过PLC
    丙丁先生 2025-01-06 09:23 51浏览
  • 随着市场需求不断的变化,各行各业对CPU的要求越来越高,特别是近几年流行的 AIOT,为了有更好的用户体验,CPU的算力就要求更高了。今天为大家推荐由米尔基于瑞芯微RK3576处理器推出的MYC-LR3576核心板及开发板。关于RK3576处理器国产CPU,是这些年的骄傲,华为手机全国产化,国人一片呼声,再也不用卡脖子了。RK3576处理器,就是一款由国产是厂商瑞芯微,今年第二季推出的全新通用型的高性能SOC芯片,这款CPU到底有多么的高性能,下面看看它的几个特性:8核心6 TOPS超强算力双千
    米尔电子嵌入式 2025-01-03 17:04 35浏览
我要评论
0