AVR c语言编程风格
扫描二维码
随时随地手机看文章
作为一个初学者如何具有良好的程序设计风格呢?我想引用一个关于初学者请教编程大师的故事让读者自己去领悟。
有一位编程大师,他写非结构化的程序,一位初学者刻意模仿他,也写非结构化的程序。当他让大师看他的进步时,大师批评了他的非结构化程序:“ 对一位编程大师合适的东西未必对一个初学者同样合适,在超越结构化之前,你必须理解编程之道。” 我个人认为作为一个初学者应该踏踏实实的打好程序设计的基础,不要急功近利,舍本逐末。我走过不少弯路,希望大家能和我一样能牢记编程大师的忠告:“对编程大师合适的东西未必对一个初学者同样合适”。
本文所描述的优秀编程风格适合于大部分语言,文章中可能提到你不是很了解的概念,没有关系,你放心的读下去,当你使用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下 的文件,后者是包含程序目录下的文件。
变量名与函数名
变量以及函数命名应该按照 尽量短 , 按需长 , 具有实际意义 。可以通过下划线或者大小写结合的方法组合动词和名词组成变量函数名。下面对比好的命名方法与不好的命名方法:
好的:Delay100us();
不好的:Yanshi();好的:init_devices();
不好的:Chengxuchushihua();好的:int temp;
不好的:int dd;
外部调用
首先在模块化程序的.h文件中定义 extern
//端口初始化 externvoidport_init(void); //T2初始化 voidtimer2_init(void); //各种参数初始化 externvoidinit_devices(void);
模块化程序的.c文件中定义函数, 不要在模块化的程序中调用程序 ,及不要出现向timer2_init();这样函数的使用,因为你以后不知道你到底什么地方调用了函数,导致程序调试难度增加。可以在定义函数的过程中调用其他函数作为函数体。
/**************************采用timer2产生波形***********************/ //PWM频率=系统时钟频率/(分频系数*2*计数器上限值)) voidtimer2_init(void) { TCCR2=0x00;//stop TCNT2=0x01;//setcount OCR2=0x66;//setcompare TCCR2=(1<
在少数几个文件中调用函数,在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 } }
宏定义
宏定义主要用于两个地方:
一是用得非常多的命令或语句,利用宏将其简化。
#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位整数关键字*/
二是利用宏定义方便的进行硬件接口操作,再程序需要修改时,只需要修改宏定义即可,而不需要满篇去找命令行,进行修改。
//PD4,PD5电机方向控制如果更改管脚控制电机方向,更改PORTD|=0x10即可。 #definemoto_en1PORTD|=0x10 #definemoto_en2PORTD|=0x20 #definemoto_uen1PORTD&=~0x10 #definemoto_uen2PORTD&=~0x20 //启动TC2定时比较和溢出 #defineTC2_ENTIMSK|=(<<1OCIE2)|(1<
关于注释
为了增加程序的可读性,方便合作者读动程序,或者程序作者在一段时间之后还能看懂程序,我们需要在程序中写 注释。
在比较特殊的函数使用或者命令调用的地方加单行注释。使用方法为:
Tbuf_putchar(c,RTbuf);//将数据加入到发送缓冲区并开中断 externvoidDelay1s(uint16n);//n<=6,whenn==7,itis1.