老古开发网首页
导航:老古开发网首页文章索引索引第2113页文章分类单片机第44页→[用PIC写高效的位移操作]
| -文章搜索 - 最新文章 - |

第2716篇:用PIC写高效的位移操作

发布时间:2006年8月2日 点击次数:1010
来源:   作者:
 

用PIC写高效的位移操作  
  
  在许多模拟串行通信中需要用位移操作。

以1-W总线的读字节为例,原厂的代码是:

unsigned char read_byte(void)
{
 unsigned char i;
 unsigned char value = 0;
 for (i = 0; i < 8; i++)
 {
  if(read_bit()) value = 0 x 01<<i;
  // reads byte in, one byte at a time and then
  // shifts it left
  delay(10); // wait for rest of timeslot
 }
 return(value);
}

虽然可以用,但编译后执行效率并不高效,这也是很多朋友认为C一定不能和汇编相比的认识提供了说法。

其实完全可以深入了解C和汇编之间的关系,写出非常高效的C代码,既有C的便利,又有汇编的效率。

首先对 for (i = 0; i < 8; i++)做手术,改成递减的形式:

for(i=8;i!=0;i--),因为CPU判断一个数是否是0(只需要一个指令),比判断一个数是多大来的快(需要3个指令)。

再对value = 0 x 01<<i;做手术。

value = 0 x 01<<i;其实是一个低水平的代码,效率低,DALLAS的工程师都是NO1,奇怪为什么会如此疏忽。

仔细研究C语言的位移操作,可以发现C总是先把标志位清0,然后再把此位移入字节中,也就是说,当前移动进字节的位一定是0。

那么,既然已经是0了,我们就只剩下一个步骤:判断总线状态是否是高来决定是否改写此位,而不需要判断总线是低的情况。

于是改写如下代码:

for(i=8;i!=0;i--){
  value>>=1;                       //先右移一位,value最高位一定是0
  if(read_bit())   value =0x80;                       //判断总线状态,如果是高,就把value的最高位置1
 }

这样一来,整个代码变得极其高效,编译后根本就是汇编级的代码。

 

再举一个例子:

在采集信号方面,经常是连续采集N次,最后求其平均值。

一般的,无论是用汇编或C,在采集次数上都推荐用8,16,32、64、128、256等次数,因为这些数都比较特殊,对于MCU计算有很大好处。

我们以128次采样为例:注:sampling()为外部采样函数。

unsigned int total;

unsigned char i,val;

for(i=0;i<128;i++){

total+=sampling();

}

val=total/128;

以上代码是很多场合都可以看见的,但是效率并不怎么样,狂浪费资源。

结合C和汇编的关系,再加上一些技巧,就可以写出天壤之别的汇编级的C代码出来

首先分析128这个数是0B10000000,发现其第7位是1,其他低位全是0,那么就可以判断第7位的状态来判断是否到了128次采样次数。

在分析除以128的运算,上面的代码用了除法运算,浪费了N多资源,完全可以用右移的方法来代替之,

val=total/128等同于val=(unsigned char)(total>>7);

再观察下去:total>>7还可以变通成(total<<1)>>8,先左移动一位,再右移动8位,不就成了右移7位了么?

可知道位移1,4,8的操作只需要一个指令哦。

有上面的概验了,就可以写出如下的代码:

unsigned int total;

unsigned char i=0

unsigned char val;

while(!(i&0x80)){                 //判断i第7位,只需要一个指令。

total+=sampling();

i++;

}

val=(unsigned char)((total<<1)>>8);                    //几个指令就代替了几十个指令的除法运算

 

哈哈,发现什么?代码量竟然可以减少一大半,运算速度可以提高几倍。

再回头,就可以理解为什么采样次数要用推荐的一些特殊值了。

 


欢迎进入老古论坛进行讨论
[单片机] 相关文章:
一个高效的模拟SPI的读写函数
简介:
一个高效的模拟SPI的读写函数。 unsigned char RW_SPI(unsigned char val){ unsigned char BitCounter; for (BitCounter=8;BitCounter!=0;BitCounter--){ CLK=0; //write DI=0; if(val&amp;0x80) ......

PIC8位在PICC中的数据类型
转:CRC算法原理及C语言实现
初浅研究PIC之延时函数和循环体优化
在PICC中使用常数指针
 
下一个:[综合电子]ADSL互通性测试简介
简介:
现代通信   随着ADSL商用化程度的日渐提高,测试重点已从严格的一致性(Conformance)测试转向了互通性(Interoperability)测试。DSL论坛的TR-048和TR-067互通性测试规范是目前业界通用的两套测试标准,以保证由不同生产厂家所生产的ADSL设备之间,在满足规定的基本数据吞吐量及性能要求条件下,实现完全的互连互通。 第一个ADSL互通性测试规范的诞生   当前,ADSL Modem基本上还处于和DSLAM捆绑销售的方式,这是在早期设......

上一个:[单片机]c51控制双色点阵左移显示与右移显示.

老古开发网版权所有 2006年9月 asp.Net V2.0 设计:老古
页面缓存:30分钟 执行时间:31毫秒