当前位置:首页 > 单片机 > 单片机
[导读] 作为一个初学者如何具有良好的程序设计风格呢?我想引用一个关于初学者请教编程大师的故事让读者自己去领悟。有一位编程大师,他写非结构化的程序,一位初学者刻意模仿他,也写非结构化的程序。当他让大

作为一个初学者如何具有良好的程序设计风格呢?我想引用一个关于初学者请教编程大师的故事让读者自己去领悟。

有一位编程大师,他写非结构化的程序,一位初学者刻意模仿他,也写非结构化的程序。当他让大师看他的进步时,大师批评了他的非结构化程序:“ 对一位编程大师合适的东西未必对一个初学者同样合适,在超越结构化之前,你必须理解编程之道。” 我个人认为作为一个初学者应该踏踏实实的打好程序设计的基础,不要急功近利,舍本逐末。我走过不少弯路,希望大家能和我一样能牢记编程大师的忠告:“对编程大师合适的东西未必对一个初学者同样合适”。


本文所描述的优秀编程风格适合于大部分语言,文章中可能提到你不是很了解的概念,没有关系,你放心的读下去,当你使用AVR一个月之后,你什么都明白了。

AVR c语言优秀编程风格

文件结构

模块化的程序应该是有一个很好的程序结构的。AVR C语言程序有两种用户文件,.c程序文件,.h头文件,程序中编写过程中需要在.c文件中包含.h头文件。初学者往往出现重复包含或者头文件包含错误的问题,我当时也时常为这种错误而发愁。下面我以我写的电机驱动例程来给大家说明一下,优秀的编程文件结构。

这个工程中有8个文件,一个说明文件,如下图:下载程序例子 电机控制案例 。

我写的成型的程序的文件个数基本上都是偶数,因为每一个结构化的函数定义.c文件都会对应一个.h文件。main.c对应config.h。我们来看看各文件的包含关系。下面我们看看这些文件的包含关系与内容:[推荐的文件包含顺序与关系]

  • 所有.c文件都包含了config.h文件。如: #include "config.h"

  • 在config.h 中有如下代码:

    #include"delay.h"
    #include"devICe_init.h"
    #include"motor.h"
  • 这样做就不容易出现错误的包含关系,为了预防万一,我们还引入了宏定义与预编译。如下:

    #ifndef_UNIT_H__
    #define_UNIT_H__1
    //100us
    externvoidDelay100us(uint8n);
    //1s
    externvoidDelay1s(uint16n);//n<=6,whenn==7,itis1.
    //1ms
    externvoidDelay1ms(uint16n);
    
    #endif
    
    第一次包含本文件的时候正确编译,并且#define_UNIT_H__1,第二次包含本文件#ifndef_UNIT_H__就不再成立,跳过文件。
    预编译还有更多的用途,比如可以根据不同的值编译不同的语句,如下:
    //#pragmaREGPARMS
    #ifCPU_TYPE==M128
    #include
    #endif
    #ifCPU_TYPE==M64
    #include
    #endif
    #ifCPU_TYPE==M32
    #include
    #endif
    #ifCPU_TYPE==M16
    #include
    #endif
    #ifCPU_TYPE==M8
    #include
    #endif
  • #include 与 #include "filename" 的区别 :前者是包含系统目录include下 的文件,后者是包含程序目录下的文件。

变量名与函数名

变量以及函数命名应该按照 尽量短 , 按需长 , 具有实际意义 。可以通过下划线或者大小写结合的方法组合动词和名词组成变量函数名。下面对比好的命名方法与不好的命名方法:

  1. 好的:Delay100us();
    不好的:Yanshi();

  2. 好的:init_devices();
    不好的:Chengxuchushihua();

  3. 好的:int temp;
    不好的:int dd;

外部调用

  1. 首先在模块化程序的.h文件中定义 extern

    //端口初始化
    externvoidport_init(void);
    
    //T2初始化
    voidtimer2_init(void);
    
    //各种参数初始化
    externvoidinit_devices(void);
  2. 模块化程序的.c文件中定义函数, 不要在模块化的程序中调用程序 ,及不要出现向timer2_init();这样函数的使用,因为你以后不知道你到底什么地方调用了函数,导致程序调试难度增加。可以在定义函数的过程中调用其他函数作为函数体。

    /**************************采用timer2产生波形***********************/
    //PWM频率=系统时钟频率/(分频系数*2*计数器上限值))
    voidtimer2_init(void)
    {
    TCCR2=0x00;//stop
    TCNT2=0x01;//setcount
    OCR2=0x66;//setcompare
    TCCR2=(1<
  3. 在少数几个文件中调用函数,在main.c中调用大部分函数,在interupts.c中根据不同的中断调用服务函数。

    voidmain(void)
    {
    
    /******************************************************************************/
    //初始工作
    /******************************************************************************/
    init_devices();
    
    while(1)
    {
    for_ward(0);				//默认速度运转正
    Delay1s(5);				//延时5s
    motor_stop();			//停止
    Delay1s(5);				//延时5s
    back_ward(0);		//默认速度运转反
    Delay1s(5);				//延时5s
    speed_add(20);			//加速
    Delay1s(5);				//延时5s
    speed_subtract(20);		//减速
    Delay1s(5);				//延时5s
    }
    
    }

宏定义

宏定义主要用于两个地方:

  1. 一是用得非常多的命令或语句,利用宏将其简化。

    #ifndefTRUE
    #defineTRUE1
    #endif
    #ifndefFALSE
    #defineFALSE0
    #endif
    #ifndefNULL
    #defineNULL0
    #endif
    #defineMIN(a,b)			((ab)?(a):(b))
    #defineABS(x)			((x>)?(x):(-x))
    typedefunsignedcharuint8;/*定义可移植的无符号8位整数关键字*/
    typedefsignedcharint8;/*定义可移植的有符号8位整数关键字*/
    typedefunsignedintuint16;/*定义可移植的无符号16位整数关键字*/
    typedefsignedintint16;/*定义可移植的有符号16位整数关键字*/
    typedefunsignedlonguint32;/*定义可移植的无符号32位整数关键字*/
    typedefsignedlongint32;/*定义可移植的有符号32位整数关键字*/
  2. 二是利用宏定义方便的进行硬件接口操作,再程序需要修改时,只需要修改宏定义即可,而不需要满篇去找命令行,进行修改。

    //PD4,PD5电机方向控制如果更改管脚控制电机方向,更改PORTD|=0x10即可。
    #definemoto_en1PORTD|=0x10
    #definemoto_en2PORTD|=0x20
    #definemoto_uen1PORTD&=~0x10
    #definemoto_uen2PORTD&=~0x20
    //启动TC2定时比较和溢出
    #defineTC2_ENTIMSK|=(<<1OCIE2)|(1<

关于注释

为了增加程序的可读性,方便合作者读动程序,或者程序作者在一段时间之后还能看懂程序,我们需要在程序中写 注释。

  1. 在比较特殊的函数使用或者命令调用的地方加单行注释。使用方法为:

    Tbuf_putchar(c,RTbuf);//将数据加入到发送缓冲区并开中断
    externvoidDelay1s(uint16n);//n<=6,whenn==7,itis1.
本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭