C语言高效编程的几招(绝对实用,绝对经典)

编写高效简洁的C语言代码,是许多软件工程师追求的目标。废话不说,走起!

第一招:以空间换时间

计算机程序中最大的矛盾是空间和时间的矛盾,那么,从这个角度出发逆向思维来考虑程序的效率问题

eg.字符串的赋值

//方法A  通常办法
#define LEN 32
char string1[LEN];
memset(string1,0,LEN);
strcpy(string1,"This is an example!!");

//方法B
const char string2[LEN]="This is an example";
char *cp;
cp=string2;//(使用的时候可以指间用指针来操作)

从上面的例子可以看出,A和B的效率是不能比的,在同样的存储空间下,B直接使用指针就可以操作了,而A需要调用两个字符函数才能完成;B的缺点在于灵活性没有A好,在需要频繁更改一个字符串内容的时候,A具有更好的灵活性;如果采用方法B,则需要预存许多字符串,虽然占用了大量的内存,但是获得了程序执行的高效率。

如果系统的实时性要求很高,内存还有一些,那我推荐使用该招数

该招数的边招,使用宏函数而不是函数;完成I/O的getchar(),做字符测试的sdigit都是得到观法认可的例子,这样做最根本的理由就是执行效率,宏可以避免函数调用的开销。

//方法C
#define bwMCDR2_ADDRESS 4
#define bsMCDR2_ADDRESS 17
int BIT_MASK(int _bf)
{
    return ((IU<<(bw## _bf))-1)    << (bs##_bf);
}
void SET_BITS(int _dst,int _bf,int _val)
{
    _dst=((_dst)&~(BIT_MASK(_bf)))|     (((_val)<<(bs##_bf)&(BIT_MASK(_bf))))
}
SET_BITS(MCDR2,MCDR2_ADDRESS,RegisterNumber);

//方法D
#define bwMCDR2_ADDRESS 4
#define bsMCDR2_ADDRESS 17
#define bmMCDR2_ADDRESS BIT_MASK(MCDR2_ADDRESS)
#define BIT_MASK(_bf)(((IU<<(bw## _bf))-1)    << (bs##_bf))
#define SET_BITS(_dst,_bf,_val)((_dst)&~(BIT_MASK(_bf)))|     (((_val)<<(bs##_bf)&(BIT_MASK(_bf))))
SET_BITS(MCDR2,MCDR2_ADDRESS,RegisterNumber);

宏函数和函数的区别在于,宏函数占用了大量的空间而函数占用了时间,大家要知道的是,函数调用是要使用系统的栈来保存数据的,如果编译器里有栈检查选项,一般在函数的头会潜入一些汇编语句对当前栈进行检查;同时,CPU也要在函数调用时保存和恢复当前的现场,进行压栈和出栈操作,所以,函数调用需要一些CPU事件,而宏函数不存在这个问题。宏函数仅仅作为预先写好的代码嵌入到当前程序,不会产生函数调用,所以仅仅是占用了空间,在频繁调用同一个宏函数时候,该现象尤其突出。

D方法是我看到的最好的置位操作函数,是ARM公司源码一部分,在短短的三行内实现了很多功能,几乎涵盖了所有位操作功能,C方法是其变体,其中滋味还需大家仔细体会!!

第2招:数学方法解决问题

现在我们演绎高效C语言编写的第二招---采用数学方法来解决问题。

数学是计算机之母,没有数学的依据和基础,就没有计算机发展,所以在编写程序的时候,采用一些数学方法会对程序的执行效率有数量级的提高。

求1~100的和

//方法E
int i,j=0;
for(i=0;i<100;i++)
{
    j=j+i;
}
//方法F
I=(100*(1+100))/2

很显然,效果自然不言而喻,所以现在我们编程序的时候,更多的是动脑筋找规律,最大限度地发挥数学的威力提高程序运行效率。

第3招:使用位操作

实现高效的C语言编写的第三招----使用位操作,减少除法和取模的运算。

在计算机程序中,数据的位是可以操作的最小数据单位,理论上可以用“位运算”来完成所有的运算和操作。一般的位操作是用来控制硬件的,或者做数据变换使用,但是,灵活的位操作可以有效提高程序运行的效率。

//方法:G
int I,J
I=257/8;
J=456%32;

//方法:H
int I,J
I=257>>3;
J=456-(456>>4>>4);

在字面上好像H比G麻烦好多,但是,仔细查看产生的汇编代码就会明白,方法G调用了基本的取模函数和除法函数,既有函数调用,还有很多汇编程序和寄存器参与运算;而方法H则仅仅是极具相关的汇编,代码更简洁,效率更高。当然,由于编译器的不同,可能效率差距不大。

运用这招需要注意的是,因为CPU的不同而产生的问题,比如说,在Pc上用这招编写的程序,并在pC上调试通过,在一直到一个16位机子平台上的时候,可能会产生代码隐患,所以只有在一定技术进阶的基础下才可以使用这招。

使用C语言进行高效编程,我的体会仅此而已,在此还请各位高手共同切磋,希望各位能给出更好的方法,大家一起提高我们的编程技巧。

转自:http://www.cnblogs.com/haoyuanyuan/p/3234200.html

时间: 2025-01-07 04:44:13

C语言高效编程的几招(绝对实用,绝对经典)的相关文章

C语言高效编程的几招,你会了几招了?

编写高效简洁的C 语言代码,是许多软件工程师追求的目标.本文就工作中的一些体会和经验做相关的阐述,不对的地方请各位指教. 第1 招:以空间换时间 计算机程序中最大的矛盾是空间和时间的矛盾,那么,从这个角度出发逆向思维来考虑程序的效率问题,我们就有了解决问题的第1 招--以空间换时间. 例如:字符串的赋值. 方法A,通常的办法: #define LEN 32 char string1 [LEN]; memset (string1,0,LEN); strcpy (string1,"This is a

[.NET] 《C# 高效编程》(一) - C# 语言习惯

C# 语言习惯 目录 一.使用属性而不是可访问的数据成员 二.使用运行时常量(readonly)而不是编译时常量(const) 三.推荐使用 is 或 as 操作符而不是强制类型转换 四.使用 Conditional 特性而不是 #if 条件编译 五.为类型提供 ToString() 方法 六.理解几个等同性判断之间的关系 七.理解 GetHashCode() 的陷阱 八.推荐使用查询语法而不是循环 九.避免在 API 中使用转换操作符 十.使用可选参数减少方法重载的数量 十一.理解短小方法的优

[.NET] 《Effective C#》快速笔记 - C# 高效编程要点补充

<Effective C#>快速笔记 - C# 高效编程要点补充 目录 四十五.尽量减少装箱拆箱 四十六.为应用程序创建专门的异常类 四十七.使用强异常安全保证 四十八.尽量使用安全的代码 四十九.实现与 CLS 兼容的程序集 五十.实现小尺寸.高内聚的程序集 这是这一系列的最后一篇. 四十五.尽量减少装箱拆箱 值类型是数据的容器,不支持多态. 装箱把一个值类型放在一个未确定类型的引用对象中,让该值作为引用类型所使用.拆箱指从引用类型的位置取出值的一个副本. 装箱拆箱都是比较影响性能的手段,应

我的高效编程秘诀

高效编程.高效学习.高效工作.高效生活 高效的意思是指在相同或更短的时间里完毕比其她人很多其它的任务,并且质量与其她人一样或者更好.用英文来说就是write less.do more.这是程序永远的主题,在我们日常编程中一段代码常常复用的时候我们会进行代码封装,借助一些工具来高速定位错误这些都有利于提高我们编程的效率.今天和小伙伴们聊聊高效编程的秘诀,事实上不仅仅是高效编程的秘诀,更是高效学习.生活和工作的秘诀,由于小编感觉编程就如同人生一样,一样的丰富多彩,一样的璀璨夺目.指导小编高效编程.生

【转载】架构师速成-如何高效编程

引子 赵云大喝一声,挺枪骤马杀入重围,左冲右突,如入无人之境.那枪浑身上下,若舞梨花:遍体纷纷,如飘瑞雪. 赵云是所有历史人物中我最喜欢的一个,如果放到现代,他走了it的道路,一定可以成为一个编程高手.为什么? 其实古时打仗也是一门技术活,需要有勇有谋,跟it没有什么太大的区别.打仗要修身.修技.修器才能左冲右突,如入无人之境,同样做it也要修身.修技.修器,才能高效编程,如入无人之境. 高效编程的修炼 何谓修身.修技.修器? 修身,古今都是修炼自己的体魄和思维,使自己体魄强健,思维敏捷 修技,

架构师速成-如何高效编程

今天看见有个csdn的征文大赛,谈谈如何高效编程,正好之前有些感触,先沉淀下来. 引子 赵云大喝一声,挺枪骤马杀入重围,左冲右突,如入无人之境.那枪浑身上下,若舞梨花:遍体纷纷,如飘瑞雪. 赵云是所有历史人物中我最喜欢的一个,如果放到现代,他走了it的道路,一定可以成为一个编程高手.为什么? 其实古时打仗也是一门技术活,需要有勇有谋,跟it没有什么太大的区别.打仗要修身.修技.修器才能左冲右突,如入无人之境,同样做it也要修身.修技.修器,才能高效编程,如入无人之境. 高效编程的修炼 何谓修身.

高效编程之指针跳转的影响

C语言的指针跳转访问是一个灵活高效的机制,但是再高效也是要花费额外时间的,下面这个程序如果在gcc -O0不优化的情况,就可以看出指针跳转比直接访问要多消耗10%左右的时间 #include <stdio.h> #include <sys/time.h> int main() { int m = 0; int *p = &m; int i =0; int n = 1; struct timeval time1, time2; m = 0; gettimeofday(&

《Effective C#》快速笔记 - C# 高效编程要点补充

目录 四十五.尽量减少装箱拆箱 四十六.为应用程序创建专门的异常类 四十七.使用强异常安全保证 四十八.尽量使用安全的代码 四十九.实现与 CLS 兼容的程序集 五十.实现小尺寸.高内聚的程序集 这是该系列的最后一篇.也许有些理论有可能会过时,我想它仍有存在的必要,人的知识水平也是一个不断成长的过程,学会站在前人的肩膀上,尝试不断的借鉴与总结. 四十五.尽量减少装箱拆箱 值类型是数据的容器,不支持多态. 装箱把一个值类型放在一个未确定类型的引用对象中,让该值作为引用类型所使用.拆箱指从引用类型的

Go语言开发(九)、Go语言并发编程

Go语言开发(九).Go语言并发编程 一.goroutine简介 1.并发与并行简介 并行(parallel):指在同一时刻,有多条指令在多个处理器上同时执行.并发(concurrency):指在同一时刻只能有一条指令执行,但多个进程指令被快速的轮换执行,使得在宏观上具有多个进程同时执行的效果,但在微观上并不是同时执行的,只是把时间分成若干段,使多个进程快速交替的执行.并行在多处理器系统中存在,而并发可以在单处理器和多处理器系统中都存在,并发能够在单处理器系统中存在是因为并发是并行的假象,并行要