历史上的今天
返回首页

历史上的今天

今天是:2025年03月05日(星期三)

正在发生

2018年03月05日 | 开发单片机程序时,初级工程师常犯的错误

2018-03-05 来源:eefocus

    这里利用一个实际发生的例子,针对初级工程师经常犯的一个小错误,或者经常要走的一个弯路,做了针对性的纠正。希望可以帮到大家,文笔不好文章中有叙述不清的地方大家多多指教。


    这篇文章我不是想说编程的规范性的东西,如果你想让自己的程序文件最起码直观的看起来美观、可读性强,推荐找华为的“C语言编程规范”。我只想说一说当我们的单片机遇到多个模块的数据需要处理,类似于“多任务”时我们应该怎么办?

 

    背景是这样的,去年9月份开始安排一个工程师开始做电动汽车交流充电桩,机械设计部分由公司机械结构部门负责。充电桩的电子部分总体上分为X个部分(用到的资源),电阻触摸屏(RS232),M1卡读写(RS232),电能计量表(RS485),语音提示(SPI),电力开关(继电器IO),通讯接口(RS485、CAN)。

    工程师做的过程非常勤奋,期间也是困难重重,改了很多个版本,总算今年6月把充电桩立起来了。

    咱们来验收一下吧,结果发现读卡的时候不能处理触摸屏,播放语音的时候不能处理读卡,语音播放不能打断或者跳跃,反正就是所有事件必须一个一个按部就班的来,一旦操作错误就需要多次执行、等待、甚至重新来过。

    一个工作3年多的工程师怎么会把产品做成这样呢?看看程序吧!

    一看不要紧,吓一跳!整个的程序是没有逻辑的,一条线就往下写……

    While(1)

    {

    //上电进入主程序 或 触发触摸屏

    //播放提示语音

    Delay();//等待播放完毕

    //读取M1卡信息

    Delay();//等待读卡数据返回

    //播放提示语音

    Delay();//等待播放完毕

    //M1卡数据交互,判定下一步操作及提示

    Delay();//等待数据处理完毕

    ……

    ……

    }

    这里说这个工程师基本上对于自己设计的产品没有任何的整体概念,或者说对自己开发的程序用到设计上会有怎样的实际效果根本就不清楚。

    他犯了几个我们在程序开发过程中最忌讳的几个问题:

    1、 delay(死等)这类函数只在应该实验室验证某个功能过程中用到,在实际的产品开发时无论是主循环while中,还是其调用的函数中,亦或是中断服务程序中绝对不可以用到。

    2、 产品设计的各个子模块之间的逻辑关系太强,例如:必须等待播音完毕才能读卡进入下一步操作等。

    我们讲,产品设计中只有各个事件处理模块间的逻辑关系弱化,才能更加灵活的进行处理。例如:两个事件A和B,如果程序开发时将A做成B事件的必要条件,B事件的触发就必须等待A事件的发生。反之如果A事件作为B事件处理的一个特殊情况,那么程序开发起来就变得灵活很多。

    3、 没有考虑到单片机本身是一个单核单任务的架构,每一个事件都会独占CPU内核,当多个任务模块同时存在时我们应该对各个事件进行区分,我们应当分情况、分事件实时性要求等区分对待。

    那么针对于这样的问题,或者是遇到类似的项目我们应该如何处理呢?

    我提几条建议:

    1、将硬件系统区分为独立单元单独做成底层驱动函数和应用函数,并且函数正常应该有参数和返回值,其中返回值是必要的。如何衡量这类函数呢?这类函数可移植性强,只要一个.h文件和一个.c文件就可以随意放到任何工程中。例如:语音播放、M1读卡、485处理等等。

    2、将1中的所有函数进行时间评估,评估点有两个。一个是函数的执行时间t,第二个是函数的周期性发生的时间T,一个最基本的条件是t < T,理想情况应该是t << T。

    3、建立一个集中逻辑处理函数,在这个函数中对1中的各个函数进行调度。这个函数发挥的作用相当于嵌入式系统中的系统调度。这种调度是整个硬件逻辑中所有事件处理的调度,它的目的是完成一个处理过程,但是绝不依赖于任意事件的必要处理过程。这样就将问题2中提到的事件间的逻辑关系弱化了,处理起来变得十分灵活,使得各个关系不在相互必要。

    4、为了保证前面内容的正常实施还需要针对各类事件的周期,建立一个必要的时间管理函数,时间函数的基础一般情况下由一个内部定时器的中断来完成,中断的周期一般我们考虑5-10ms。按照实际需求将N个定时器中断定义为一个事件处理的周期TT,这个周期应该保证处理完最恶劣情况可能发生的所有t,且保证TT < T。

    5、 这其中也有例外,一些实时性要求高的事件应当用中断完成。其中中断处理函数的处理事件应尽量短,时间要求参见2。


推荐阅读

史海拾趣

百蓉(ECE)公司的发展小趣事

随着公司的发展,百蓉公司意识到技术创新是保持竞争力的关键。他们决定投入更多资源在研发上,特别是集成电路设计领域。经过几年的努力,百蓉成功开发出了一系列高效能、低功耗的集成电路芯片,广泛应用于消费电子、汽车电子和工业控制等领域。这些芯片在市场上大受欢迎,进一步提升了百蓉在电子行业中的地位。

Ho Chien Electronics Group Inc公司的发展小趣事

百蓉公司不满足于仅在国内市场发展,开始积极拓展国际市场。他们参加了多个国际电子展览,与多家国际知名企业建立了合作关系。通过深入了解不同国家和地区的市场需求,百蓉针对性地推出了符合当地消费者需求的产品,逐渐在国际市场上占据了一席之地。

Henkel公司的发展小趣事

百蓉公司不满足于仅在国内市场发展,开始积极拓展国际市场。他们参加了多个国际电子展览,与多家国际知名企业建立了合作关系。通过深入了解不同国家和地区的市场需求,百蓉针对性地推出了符合当地消费者需求的产品,逐渐在国际市场上占据了一席之地。

Futaba Electric Co Ltd公司的发展小趣事

在电子行业的激烈竞争中,百蓉(ECE)公司应运而生。创立初期,公司仅有几个热爱电子技术的年轻人,他们在一个小小的办公室里,凭借对电子技术的深厚兴趣和专业知识,开始研发新产品。那个时代,智能手机刚刚开始兴起,他们敏锐地捕捉到了触摸屏技术的巨大市场潜力。于是,他们投入大量时间和精力,研发出了一款高性能、低成本的触摸屏控制器,这一创新产品迅速在市场上获得了成功,为百蓉公司的发展奠定了坚实的基础。

DAICO公司的发展小趣事

DAICO公司成立于XXXX年,最初是一家位于加利福尼亚的小型私营公司。在公司成立的初期,DAICO专注于研发和生产中频/射频和微波控制产品,以及放大器。这些产品主要用于国防电子和航空航天领域,为DAICO在电子行业中打下了坚实的基础。

扬州国芯(Gcore)公司的发展小趣事

DAICO公司成立于XXXX年,最初是一家位于加利福尼亚的小型私营公司。在公司成立的初期,DAICO专注于研发和生产中频/射频和微波控制产品,以及放大器。这些产品主要用于国防电子和航空航天领域,为DAICO在电子行业中打下了坚实的基础。

问答坊 | AI 解惑

谁知道EWB中默认的ADC是哪个型号的芯片啊?

本帖最后由 dontium 于 2015-1-23 13:26 编辑 谁知道EWB中默认的ADC是哪个型号的芯片啊?能告诉我吗?谢谢,我做设计用那个仿真结果很好,可是不知道那具体是哪个芯片啊 …

查看全部问答>

C51使用技巧及实战.rar

C51使用技巧及实战.rar…

查看全部问答>

扩展串口芯片驱动的问题

PXA270+WinCE6.0  通过总线扩展串口 芯片 ST16c554 扩出了4个串口 对串口了解不够,请问一下,写扩展串口的驱动需要注意哪些? 对比芯片资料发现里面寄存器的配置与16550差别不大,是不是可以继承16550的pdd层? mdd层需要修改吗,还 ...…

查看全部问答>

电子信息工程的女生该选择什么样的发展方向

   我是电子信息工程的在校大三女生,我想请教一下各位,我们这个专业的女生应该选择怎样的发展方向,或深入学些什么知识,明年毕业才能找到较理想的工作?本人的动手能力和编程能力都很一般,没有深入学习。有人提议我选择单片机,但是 ...…

查看全部问答>

程序编译出现问题

请版主及各位朋友看看,我用定时器1做脉冲信号的输出,程序编译时候TIM1出现了好多错误,但是我明明已经把stm32f10x_tim.c这个文件加入到了工程项里面,不知道为什么还会出现这些问题,请大家指点一下问题出在哪里。 下载 (58.14 ...…

查看全部问答>

纳米测量中屏蔽罩的重要作用

电缆的不当使用会造成测量时间过长的问题。共轴电缆提供了一个传输信号的内导体和屏蔽。内导体和屏蔽之间存在着可供漏电流流过的旁路电阻和电容通路[1](图4)。除了作为漏电流的通路之外,旁路的R和C还构成了一个RC电路,该电路将大大放慢弱电流 ...…

查看全部问答>

TI Cortex-A8 TQ_AM335X的简介

TQ_AM335X开发板是广州天嵌计算机科技有限公司专门针对有一定开发经验或企业用户提供的一整套的学习和解决方案。TQ_AM335X提供了一个性能稳定及成本低廉的Cortex-A8开发平台,以便您快速熟悉TI AM335X处理器特性和设计方法,并加快您的产品开发进度 ...…

查看全部问答>

2440学习记录(十四)守护进程的创建

这一节学习守护进程的创建 参考文献 blog.sina.com.cn/s/blog_643d3d780100lbsg.html blog.csdn.net/mybelief321/article/details/9069659 守护进程是脱离于终端并且在后台运行的进程。守护进程脱离于终端是为了避免进程在执行过程中的信息 ...…

查看全部问答>

LPC1500体验+SCT_1、寄存器

本帖最后由 freebsder 于 2014-8-15 21:59 编辑 1, CONFIG寄存器中的UNIFY位设置timer作为1个32位还是2个16位计时器。在访问其他寄存器之前设置这个位。       UNIFY = 1: Only one register     is ...…

查看全部问答>