ImageVerifierCode 换一换
格式:DOCX , 页数:27 ,大小:103.47KB ,
资源ID:4865736      下载积分:3 金币
快捷下载
登录下载
邮箱/手机:
温馨提示:
快捷下载时,用户名和密码都是您填写的邮箱或者手机号,方便查询和重复下载(系统自动生成)。 如填写123,账号就是123,密码也是123。
特别说明:
请自助下载,系统不会自动发送文件的哦; 如果您已付费,想二次下载,请登录后访问:我的下载记录
支付方式: 支付宝    微信支付   
验证码:   换一换

加入VIP,免费下载
 

温馨提示:由于个人手机设置不同,如果发现不能下载,请复制以下地址【https://www.bdocx.com/down/4865736.html】到电脑端继续下载(重复下载不扣费)。

已注册用户请登录:
账号:
密码:
验证码:   换一换
  忘记密码?
三方登录: 微信登录   QQ登录  

下载须知

1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。
2: 试题试卷类文档,如果标题没有明确说明有答案则都视为没有答案,请知晓。
3: 文件的所有权益归上传用户所有。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 本站仅提供交流平台,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

版权提示 | 免责声明

本文(C语言嵌入式系统编程修炼.docx)为本站会员(b****3)主动上传,冰豆网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知冰豆网(发送邮件至service@bdocx.com或直接QQ联系客服),我们立即给予删除!

C语言嵌入式系统编程修炼.docx

1、C语言嵌入式系统编程修炼C语言嵌入式系统编程修炼(性能优化) 使用宏定义在C语言中,宏是产生内嵌代码的唯一方法。对于嵌入式系统而言,为了能达到性能要求,宏是一种很好的代替函数的方法。写一个标准宏MIN ,这个宏输入两个参数并返回较小的一个:错误做法:#define MIN(A,B) ( A = B ? A : B )正确做法:#define MIN(A,B) (A)= (B) ? (A) : (B) )对于宏,我们需要知道三点:(1)宏定义像函数;(2)宏定义不是函数,因而需要括上所有参数;(3)宏定义可能产生副作用。下面的代码:least = MIN(*p+, b);将被替换为:( (*p+

2、) = (b) ?(*p+):(b) )发生的事情无法预料。因而不要给宏定义传入有副作用的参数。 使用寄存器变量当对一个变量频繁被读写时,需要反复访问内存,从而花费大量的存取时间。为此,C语言提供了一种变量,即寄存器变量。这种变量存放在 CPU的寄存器中,使用时,不需要访问内存,而直接从寄存器中读写,从而提高效率。寄存器变量的说明符是register。对于循环次数较多的循环控制变量及循环体内反复使用的变量均可定义为寄存器变量,而循环计数是应用寄存器变量的最好候选者。(1) 只有局部自动变量和形参才可以定义为寄存器变量。因为寄存器变量属于动态存储方式,凡需要采用静态存储方式的量都不能定义为寄存器

3、变量,包括:模块间全局变量、模块内全局变量、局部static变量;(2) register是一个建议型关键字,意指程序建议该变量放在寄存器中,但最终该变量可能因为条件不满足并未成为寄存器变量,而是被放在了存储器中,但编译器中并不报错(在C+语言中有另一个建议型关键字:inline)。下面是一个采用寄存器变量的例子:/* 求1+2+3+.+n的值 */WORD Addition(BYTE n)register i,s=0;for(i=1;i外部同步RAM外部异步RAMFLASH/ROM对于程序代码,已经被烧录在FLASH或ROM中,我们可以让CPU直接从其中读取代码执行,但通常这不是一个好办法,

4、我们最好在系统启动后将FLASH或ROM中的目标代码拷贝入RAM中后再执行以提高取指令速度;对于UART等设备,其内部有一定容量的接收BUFFER,我们应尽量在BUFFER被占满后再向CPU提出中断。例如计算机终端在向目标机通过RS-232传递数据时,不宜设置UART只接收到一个BYTE就向CPU提中断,从而无谓浪费中断处理时间;如果对某设备能采取DMA方式读取,就采用DMA读取,DMA读取方式在读取目标中包含的存储信息较大时效率较高,其数据传输的基本单位是块,而所传输的数据是从设备直接送入内存的(或者相反)。DMA方式较之中断驱动方式,减少了CPU 对外设的干预,进一步提高了CPU与外设的并

5、行操作程度。 活用位操作使用C语言的位操作可以减少除法和取模的运算。在计算机程序中数据的位是可以操作的最小数据单位,理论上可以用位运算来完成所有的运算和操作,因而,灵活的位操作可以有效地提高程序运行的效率。举例如下:/* 方法1 */int i,j;i = 879 / 16;j = 562 % 32;/* 方法2 */int i,j;i = 879 4;j = 562 - (562 5 5);对于以2的指数次方为*、/或%因子的数学运算,转化为移位运算通常可以提高算法效率。因为乘除运算指令周期通常比移位运算大。C语言位运算除了可以提高运算效率外,在嵌入式系统的编程中,它的另一个最典型的应用,而

6、且十分广泛地正在被使用着的是位间的与(&)、或(|)、非()操作,这跟嵌入式系统的编程特点有很大关系。我们通常要对硬件寄存器进行位设置,譬如,我们通过将AM186ER型 80186处理器的中断屏蔽控制寄存器的第低6位设置为0(开中断2),最通用的做法是:#define INT_I2_MASK 0x0040wTemp = inword(INT_MASK);outword(INT_MASK, wTemp &INT_I2_MASK);而将该位设置为1的做法是:#define INT_I2_MASK 0x0040wTemp = inword(INT_MASK);outword(INT_MASK, wT

7、emp | INT_I2_MASK);判断该位是否为1的做法是:#define INT_I2_MASK 0x0040wTemp = inword(INT_MASK);if(wTemp & INT_I2_MASK) /* 该位为1 */上述方法在嵌入式系统的编程中是非常常见的,我们需要牢固掌握。 总结在性能优化方面永远注意80-20准备,不要优化程序中开销不大的那80%,这是劳而无功的。宏定义是C语言中实现类似函数功能而又不具函数调用和返回开销的较好方法,但宏在本质上不是函数,因而要防止宏展开后出现不可预料的结果,对宏的定义和使用要慎而处之。很遗憾,标准C至今没有包括C+中inline函数的功能

8、,inline函数兼具无调用开销和安全的优点。使用寄存器变量、内嵌汇编和活用位操作也是提高程序效率的有效方法。除了编程上的技巧外,为提高系统的运行效率,我们通常也需要最大可能地利用各种硬件设备自身的特点来减小其运转开销,例如减小中断次数、利用DMA传输方式等。C语言嵌入式系统编程修炼(键盘操作) 处理功能键功能键的问题在于,用户界面并非固定的,用户功能键的选择将使屏幕画面处于不同的显示状态下。例如,主画面如图1:图1 主画面当用户在设置XX上按下Enter键之后,画面就切换到了设置XX的界面,如图2:图2 切换到设置XX画面程序如何判断用户处于哪一画面,并在该画面的程序状态下调用对应的功能键处

9、理函数,而且保证良好的结构,是一个值得思考的问题。让我们来看看WIN32编程中用到的窗口概念,当消息(message)被发送给不同窗口的时候,该窗口的消息处理函数(是一个callback函数)最终被调用,而在该窗口的消息处理函数中,又根据消息的类型调用了该窗口中的对应处理函数。通过这种方式,WIN32有效的组织了不同的窗口,并处理不同窗口情况下的消息。我们从中学习到的就是:(1)将不同的画面类比为WIN32中不同的窗口,将窗口中的各种元素(菜单、按钮等)包含在窗口之中;(2)给各个画面提供一个功能键消息处理函数,该函数接收按键信息为参数;(3)在各画面的功能键消息处理函数中,判断按键类型和当前

10、焦点元素,并调用对应元素的按键处理函数。/* 将窗口元素、消息处理函数封装在窗口中 */struct windowsBYTE currentFocus;ELEMENT elementELEMENT_NUM;void (*messageFun) (BYTE keyValue);/* 消息处理函数 */void messageFunction(BYTE keyValue)BYTE i = 0;/* 获得焦点元素 */while ( (element i.ID!= currentFocus)& (i ELEMENT_NUM) )i+;/* 消息映射 */if(i ELEMENT_NUM)switch

11、(keyValue)case OK:elementi.OnOk();break;在窗口的消息处理函数中调用相应元素按键函数的过程类似于消息映射,这是我们从WIN32编程中学习到的。编程到了一个境界,很多东西都是相通的了。其它地方的思想可以拿过来为我所用,是为编程中的拿来主义。在这个例子中,如果我们还想玩得更大一点,我们可以借鉴MFC中处理MESSAGE_MAP的方法,我们也可以学习MFC定义几个精妙的宏来实现消息映射。 处理数字键用户输入数字时是一位一位输入的,每一位的输入都对应着屏幕上的一个显示位置(x坐标,y坐标)。此外,程序还需要记录该位置输入的值,所以有效组织用户数字输入的最佳方式是定

12、义一个结构体,将坐标和数值捆绑在一起:/* 用户数字输入结构体 */typedef struct tagInputNumBYTE byNum; /* 接收用户输入赋值 */BYTE xPos; /* 数字输入在屏幕上的显示位置x坐标 */BYTE yPos; /* 数字输入在屏幕上的显示位置y坐标 */InputNum, *LPInputNum;那么接收用户输入就可以定义一个结构体数组,用数组中的各位组成一个完整的数字:InputNum inputElementNUM_LENGTH; /* 接收用户数字输入的数组 */* 数字按键处理函数 */extern void onNumKey(BYTE

13、 num)if(num=0| num=1) /* 只接收二进制输入 */* 在屏幕上显示用户输入 */DrawText(inputElementcurrentElementInputPlace.xPos, inputElementcurrentElementInputPlace.yPos, %1d, num);/* 将输入赋值给数组元素 */inputElementcurrentElementInputPlace.byNum = num;/* 焦点及光标右移 */moveToRight();将数字每一位输入的坐标和输入值捆绑后,在数字键处理函数中就可以较有结构的组织程序,使程序显得很紧凑。整理

14、用户输入继续第2节的例子,在第2节的onNumKey函数中,只是获取了数字的每一位,因而我们需要将其转化为有效数据,譬如要转化为有效的XXX数据,其方法是:/* 从2进制数据位转化为有效数据:XXX */void convertToXXX()BYTE i;XXX = 0;for (i = 0; i NUM_LENGTH; i+)XXX += inputElementi.byNum*power(2, NUM_LENGTH - i - 1);反之,我们也可能需要在屏幕上显示那些有效的数据位,因为我们也需要能够反向转化:/* 从有效数据转化为2进制数据位:XXX */void convertFrom

15、XXX()BYTE i;XXX = 0;for (i = 0; i NUM_LENGTH; i+)inputElementi.byNum = XXX / power(2, NUM_LENGTH - i - 1) % 2;当然在上面的例子中,因为数据是2进制的,用power函数不是很好的选择,直接用移位操作效率更高,我们仅是为了说明问题的方便。试想,如果用户输入是十进制的,power函数或许是唯一的选择了。总结本篇给出了键盘操作所涉及的各个方面:功能键处理、数字键处理及用户输入整理,基本上提供了一个全套的按键处理方案。对于功能键处理方法,将LCD屏幕与Windows窗口进行类比,提出了较新颖地解

16、决屏幕、键盘繁杂交互问题的方案。计算机学的许多知识都具有相通性,因而,不断追赶时髦技术而忽略基本功的做法是徒劳无意的。我们最多需要精通三种语言(精通,一个在如今的求职简历里泛滥成灾的词语),最佳拍档是汇编、C、C+(或JAVA),很显然,如果你精通了这三种语言,其它语言你应该是可以很快熟悉的,否则你就没有精通它们。C语言嵌入式系统编程修炼(屏幕操作) 汉字处理现在要解决的问题是,嵌入式系统中经常要使用的并非是完整的汉字库,往往只是需要提供数量有限的汉字供必要的显示功能。例如,一个微波炉的LCD上没有必要提供显示电子邮件的功能;一个提供汉字显示功能的空调的LCD上不需要显示一条短消息,诸如此类。

17、但是一部手机、小灵通则通常需要包括较完整的汉字库。如果包括的汉字库较完整,那么,由内码计算出汉字字模在库中的偏移是十分简单的:汉字库是按照区位的顺序排列的,前一个字节为该汉字的区号,后一个字节为该字的位号。每一个区记录94个汉字,位号则为该字在该区中的位置。因此,汉字在汉字库中的具体位置计算公式为:94*(区号-1)+位号-1。减1是因为数组是以0为开始而区号位号是以1为开始的。只需乘上一个汉字字模占用的字节数即可,即:(94*(区号-1)+位号-1)*一个汉字字模占用字节数,以16*16点阵字库为例,计算公式则为:(94*(区号-1)+(位号- 1)*32。汉字库中从该位置起的32字节信息记

18、录了该字的字模信息。对于包含较完整汉字库的系统而言,我们可以以上述规则计算字模的位置。但是如果仅仅是提供少量汉字呢?譬如几十至几百个?最好的做法是:定义宏:# define EX_FONT_CHAR(value)# define EX_FONT_UNICODE_VAL(value) (value),# define EX_FONT_ANSI_VAL(value) (value),定义结构体:typedef struct _wide_unicode_font16x16WORD value; /* 内码 */BYTE data32; /* 字模点阵 */Unicode;#define CHINES

19、E_CHAR_NUM /* 汉字数量 */字模的存储用数组:Unicode chineseCHINESE_CHAR_NUM =EX_FONT_CHAR(业)EX_FONT_UNICODE_VAL(0x4e1a)0x04, 0x40, 0x04, 0x40, 0x04, 0x40, 0x04, 0x44, 0x44, 0x46, 0x24, 0x4c, 0x24, 0x48, 0x14, 0x50, 0x1c, 0x50, 0x14, 0x60, 0x04, 0x40, 0x04, 0x40, 0x04, 0x44, 0xff, 0xfe, 0x00, 0x00, 0x00, 0x00,EX_F

20、ONT_CHAR(中)EX_FONT_UNICODE_VAL(0x4e2d)0x01, 0x00, 0x01, 0x00, 0x21, 0x08, 0x3f, 0xfc, 0x21, 0x08, 0x21, 0x08, 0x21, 0x08, 0x21, 0x08, 0x21, 0x08,0x3f, 0xf8, 0x21, 0x08, 0x01, 0x00, 0x01, 0x00, 0x01, 0x00, 0x01, 0x00, 0x01, 0x00,EX_FONT_CHAR(云)EX_FONT_UNICODE_VAL(0x4e91)0x00, 0x00, 0x00, 0x30, 0x3f,

21、0xf8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0c, 0xff, 0xfe, 0x03, 0x00, 0x07, 0x00,0x06, 0x40, 0x0c, 0x20, 0x18, 0x10, 0x31, 0xf8, 0x7f, 0x0c, 0x20, 0x08, 0x00, 0x00,EX_FONT_CHAR(件)EX_FONT_UNICODE_VAL(0x4ef6)0x10, 0x40, 0x1a, 0x40, 0x13, 0x40, 0x32, 0x40, 0x23, 0xfc, 0x64, 0x40, 0xa4, 0x40, 0x28, 0x40, 0x

22、2f, 0xfe,0x20, 0x40, 0x20, 0x40, 0x20, 0x40, 0x20, 0x40, 0x20, 0x40, 0x20, 0x40, 0x20, 0x40要显示特定汉字的时候,只需要从数组中查找内码与要求汉字内码相同的即可获得字模。如果前面的汉字在数组中以内码大小顺序排列,那么可以以二分查找法更高效的查找到汉字的字模。这是一种很有效的组织小汉字库的方法,它可以保证程序有很好的结构。系统时间显示从NVRAM中可以读取系统的时间,系统一般借助NVRAM产生的秒中断每秒读取一次当前时间并在LCD上显示。关于时间的显示,有一个效率问题。因为时间有其特殊性,那就是60秒才有一

23、次分钟的变化,60分钟才有一次小时变化,如果我们每次都将读取的时间在屏幕上完全重新刷新一次,则浪费了大量的系统时间。一个较好的办法是我们在时间显示函数中以静态变量分别存储小时、分钟、秒,只有在其内容发生变化的时候才更新其显示。extern void DisplayTime()static BYTE byHour,byMinute,bySecond;BYTE byNewHour, byNewMinute, byNewSecond;byNewHour = GetSysHour();byNewMinute = GetSysMinute();byNewSecond = GetSysSecond();i

24、f(byNewHour!= byHour) /* 显示小时 */byHour = byNewHour;if(byNewMinute!= byMinute) /* 显示分钟 */byMinute = byNewMinute;if(byNewSecond!= bySecond) /* 显示秒钟 */bySecond = byNewSecond;这个例子也可以顺便作为C语言中static关键字强大威力的证明。当然,在C+语言里,static具有了更加强大的威力,它使得某些数据和函数脱离对象而成为类的一部分,正是它的这一特点,成就了软件的无数优秀设计。 动画显示动画是无所谓有,无所谓无的,静止的画面走

25、的路多了,也就成了动画。随着时间的变更,在屏幕上显示不同的静止画面,即是动画之本质。所以,在一个嵌入式系统的LCD上欲显示动画,必须借助定时器。没有硬件或软件定时器的世界是无法想像的:(1) 没有定时器,一个操作系统将无法进行时间片的轮转,于是无法进行多任务的调度,于是便不再成其为一个多任务操作系统;(2) 没有定时器,一个多媒体播放软件将无法运作,因为它不知道何时应该切换到下一帧画面;(3) 没有定时器,一个网络协议将无法运转,因为其无法获知何时包传输超时并重传之,无法在特定的时间完成特定的任务。因此,没有定时器将意味着没有操作系统、没有网络、没有多媒体,这将是怎样的黑暗?所以,合理并灵活地

26、使用各种定时器,是对一个软件人的最基本需求!在80186为主芯片的嵌入式系统中,我们需要借助硬件定时器的中断来作为软件定时器,在中断发生后变更画面的显示内容。在时间显示xx:xx中让冒号交替有无,每次秒中断发生后,需调用ShowDot:void ShowDot()static BOOL bShowDot = TRUE; /* 再一次领略static关键字的威力 */if(bShowDot)showChar(:,xPos,yPos);elseshowChar( ,xPos,yPos);bShowDot = ! bShowDot;菜单操作无数人为之绞尽脑汁的问题终于出现了,在这一节里,我们将看到,

27、在C语言中哪怕用到一丁点的面向对象思想,软件结构将会有何等的改观!笔者曾经是个笨蛋,被菜单搞晕了,给出这样的一个系统:图1 菜单范例要求以键盘上的 键切换菜单焦点,当用户在焦点处于某菜单时,若敲击键盘上的OK、CANCEL键则调用该焦点菜单对应之处理函数。我曾经傻傻地这样做着:/* 按下OK键 */void onOkKey()/* 判断在什么焦点菜单上按下Ok键,调用相应处理函数 */Switch(currentFocus)case MENU1:menu1OnOk();break;case MENU2:menu2OnOk();break;/* 按下Cancel键 */void onCancelKey()/* 判断在什么焦点菜单上按下Cancel键,调用相应处理函数 */Switch(currentFocus)case MENU1:menu1OnCancel();break;case MENU2:menu2OnCancel();break;终于有一天,我这样做了:/* 将菜单的属性和操作封装在一起 */typedef struct tagSysMenuchar *text; /* 菜单的文本 */BYTE xPos; /* 菜单在LCD上

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1