原创 提高C语言代码效率的方法

2013-3-9 12:18 1391 7 8 分类: MCU/ 嵌入式

 

       代码效率包括两个方面内容:代码的大小和代码执行速度。如果代码精简和执行速度快,我们就说这个代码效率高。一般情况下,代码精简了速度也相应提上来了。单片机的ROM和RAM的空间都很有限,当您编程时遇到单片机的ROM和RAM的不够用的时候,或者您的程序要求较高的执行速度时,我们就得面对解决代码效率问题了。如何提高代码效率?现笔者以一个LED闪烁的程序为例与您探讨。

 #i nclude<reg52.h>//包含头文件

 sbit led=P2^0;//定义位变量led,使其关联单片机管脚P2.0

 void Delayms(unsigned int t);//定义延时函数

 int main(void)//主函数(C语言程序入口函数)

 {

       while(1)

       {

             led=0;//P2.0拉低,点亮LED

             Delayms(500);//调用延时函数,延时500毫秒

             led=1;//P2.0拉高,熄灭LED

             Delayms(500);//调用延时函数,延时500毫秒

       }

       return 0;

 }

void Delayms(unsigned int t)//延时函数

{

        unsigned int i,j;

        for(i=0;i<t;i++)

               for(j=0;j<120;j++);//大约延时1毫秒

}

这是指示灯LED闪烁的C源码,这个源码在Keil uVision4 生成的程序代码是67个字节。下面我们就采用几个方法来提高这个程序的效率。

一.尽量定义局部变量

单片机程序的全局变量一般是放在通用数据存储器(RAM)中,而局部变量一般是放在特殊功能寄存器当中。处理寄存器数据的速度比处理RAM数据要快,如果在一个局部函数里调用一个全局变量将会多生成好几个代码出来。所以,少定义全局变量,多定义局部变量。如上例中,如果把延时函数里的i和j定义为全局变量,编译后程序代码会增加到79个字节,多了12个字节。

二.省略函数定义

在一个单片机程序里我们习惯在main函数的前面先定义被调用函数,然后在mian函数的下面再实现被调用函数。这样的写法固然是一个好习惯,但每定义一个函数会增加几个代码,而且函数形参数据类型越大、形参越多增加的代码就越多,显然这不是什么好事。如果不定义编译器又报错,怎么办?C编译器的编译顺序是从上往下编译,只要被调用的函数在主调函数调用之前实现就没有问题了。所以,笔者的习惯写法是不用定义函数,但要按先后顺序(被调用函数一定要在主调函数之前写好)来写函数实现,到最后再写main函数。这样做编译器不但不会报错,而且代码得到精简了。如上例中,把延时函数的定义删除了,然后把延时函数的实现搬到main函数的上面,编译后程序代码减少到63个字节,减少了4个字节。

三.省略函数形参

函数带形参,是为了在函数调用时传递实参,不但可以避免重复代码出现,还可以通过传递不同的实参值多次调用函数且实现不同的函数功能,总体代码也会得到精简。在实际编程的时候,我们只要注意,还可以进一步精简代码。对于不是多次调用或者多次调用但实参值不变的函数我们可以省略函数形参。如上例中的延时函数,我们把它改成不带形参的函数:

void Delayms()//延时函数

 {

     unsigned int i,j;

     for(i=0;i<500;i++)

          for(j=0;j<120;j++);//大约延时1毫秒

 }

编译后,程序代码变成了56个字节,精简了11个字节。

四.改换运算符

也许您可能没有注意到C运算符的运用也会影响程序代码的数量。如上例中,把延时函数里的自加运算符改成自减运算符后,如:

void Delayms(unsigned int t)//延时函数

 {

     unsigned int i,j;

     for(i=t;i>0;i--)

          for(j=120;j>0;j--);//大约延时1毫秒

 }

编译后,程序代码变成了65个字节,精简了2个字节。

通过改换运算符能达到精简代码的例子还有:

1.把求余运算表达式改为位与运算表达式。如:b=a%8 可以改为:b=a&7。

2.把乘法运算表达式改为左移运算表达式。如:b=a*8 可以改为:b=a<<3。

3.把除法运算表达式改为右移运算表达式。如:b=a/8 可以改为:b=a>>3。

五.选择合适的数据类型

C语言里选择变量的数据类型很讲究,变量的数据类型过小满足不了程序的要求,变量的数据类型过大会占用太多的RAM资源。您可能还没有注意到数据类型定义也影响程序代码的大小,而且这个影响还不小。如上例中,延时函数里的局部变量j定义的数据类型明显偏大,如果把它由unsigned int改成unsigned char 。编译后,程序代码变成了59个字节,精简了8个字节。

六.直接嵌入代码

    在您的程序里如果某个函数只调用一次,而您又要求代码提高执行速度,建议您不要采用调用函数的形式,而应该将该函数里的代码直接嵌入主调函数里,代码执行效率会大大提高。

七.使用效率高的C语句

C语言里有一个三目运算符“?”,俗称“问号表达式”。很多程序员都很喜欢使用,因为它逻辑清晰表达简洁。

看这个问号表达式:c=(a>b) ? a+1 : b+1;实际上等效于以下的if…else结构:

if (a>b)  c=a+1;

else  c=b+1;

可以看到,使用问号表达式,语句相当简洁,但它的执行效率却很低,远没有if…else语句效率高。所以,当您的程序要求提高执行速度的话,建议您不要使用问号表达式了。

另外,do…while语句也比while语句的效率高。

代码的效率问题,不是我们编程中的主要问题,除了程序要求较高的执行速度或者单片机的ROM和RAM不够用的时候才会考虑。一般情况下,我们不用在乎。如果您一味追求高效率的代码,可能会影响代码的可读性和可维护性。

转载:http://blog.21ic.com/user1/5877/archives/2012/91570.html

 

PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户422869 2013-3-13 12:44

疑惑,我记得我当时做实验的时候,双目运算记得比if……else……的函数精简呢

相关推荐阅读
XQSir 2013-08-24 23:02
三极管开关电路
今天弄了个时钟采集系统!时钟过高时出现了诸多问题,系统由3个MCU与一片CPLD构成!CPLD时钟由10MHz峰值3.3V正弦波输入,MCU采集实时时间! 问题来了,首先要对时钟源整形提供3....
XQSir 2013-07-30 11:58
485自收发
485自收发设计,两控制端短接更好! ...
XQSir 2013-06-23 17:21
MSP430编程器仿真器以及JTAG、SBW、BSL接口的区别
对于51系统来说,很容易理解编程器和仿真器。 通俗的说,仿真器是用来调试仿真的,编程器是用来批量生产时对MCU进行烧写目标代码的。 对于MSP430来说,无论仿真还是烧写程序一般可以通过:JT...
XQSir 2013-04-04 20:01
小试S3C2410!
    今天闲来无事,不想再编烦人的论文,看到旁边的ARM9实验板,便借来玩下。找来编译器和JTAG。装了下环境算是可以搭起来了!跑了个LCD,稍加更改,效果还好!运行中有几个问题,给自己留点纪念...
XQSir 2013-03-27 15:01
基于IAP的STM32程序更新技术
    摘要:         嵌入式系统的开发最终需要将编译好的代码下载到具体的微控制器芯片上,而不同厂家的微控制器芯片有不同的下载方式。随着技术的发展和应用需求...
XQSir 2013-03-18 09:28
有关运算放大器指标
运放的指标:  1、开环差模电压增益Avd:  运放在没有外部反馈作用时的差模直流电压增益称为开环差模电压增益,它是决定运放电路运算精度的重要因素,  定义为运放开环是的输出电压与差模输入电...
EE直播间
更多
我要评论
1
7
关闭 站长推荐上一条 /3 下一条