分享

51黑:误人子弟的程序分析

 zsok 2016-01-16


51黑电子论坛:www.51hei.com/bbs/ 精选文章:

下面程序是我从别的地方搞来的,最后发现很多错误。为了避免误人子弟,特地作了一下分析,开头部分for分析是我写的,程序中红色的部分我是加上去的。
在软件仿真环境下,把晶振改成12.0MhzC51标签代码优化设为0,见图附件,测得for语句的延时为:

for(time=0;time<1;time++);>

for(time=0;time<2;time++);>

for(time=0;time<3;time++);>

for(time=0;time<4;time++);>

for(time=0;time<5;time++);>

for(time=0;time<6;time++);>

for(time=0;time<7;time++);>

for(time=0;time<8;time++);>

for(time=0;time<9;time++);>

for(time=0;time<10;time++);>

for(time=0;time<20;time++);>

for(time=0;time<60ime++);>

for(time=0;time<70ime++);>

for(time=0;time<80ime++);>

for(time=0;time<100;time++);>

for(time=0;time<200;time++);>

由上可看出,成等差数列,公差d=8。一般地,如果n>man=am+8*(n-m)




/*****************************************************

函数功能:将DS18B20传感器初始化,读取应答信号

出口参数:flag

***************************************************/

bit Init_DS18B20(void)

{

bitflag; //储存DS18B20是否存在的标志,flag=0,表示存在;flag=1,表示不存在

DQ= 1; //先将数据线拉高

for(time=0;time<2;time++);>略微延时约6微秒。实际是延时23us

DQ= 0; //再将数据线从高拉低,要求保持480~960us

for(time=0;time<200;time++);>略微延时约600微秒,以向DS18B20发出一持续480~960us的低电平复位脉冲。际是延时1607us,重大错误,都超过960us了。

DQ= 1; //释放数据线(将数据线拉高)

for(time=0;time<10;time++);>延时约30us(释放总线后需等待15~60usDS18B20输出存在脉冲)。实际是延时87us,延时多过60us,反而能保证读到存在脉冲,这点误撞了。

flag=DQ; //让单片机检测是否输出了存在脉冲(DQ=0表示存在)

for(time=0;time<200;time++);>延时足够长时间,等待存在脉冲输出完毕

return(flag); //返回检测成功标志

}

/*****************************************************

函数功能:从DS18B20读取一个字节数据

出口参数:dat

***************************************************/

unsigned char ReadOneChar(void)

{

unsignedchar i=0;

unsignedchar dat; //储存读出的一个字节数据

for(i=0;i<>

{

DQ=1; //先将数据线拉高

_nop_(); //等待一个机器周期

DQ= 0; //单片机从DS18B20读书据时,将数据线从高拉低即启动读时序

dat>>=1;

_nop_(); //等待一个机器周期

DQ= 1; //将数据线'人为'拉高,为单片机检测DS18B20的输出电平作准备

for(time=0;time<2;time++);>延时约6us,使主机在15us内采样。实际是延时23us,超过了15us

if(DQ==1)

dat|=0x80; //如果读到的数据是1,则将1存入dat

else

dat|=0x00;/如果读到的数据是0,则将0存入dat,将单片机检测到的电平信号DQr

for(time=0;time<8;time++);>延时3us,两个读时序之间必须有大于1us的恢复期。实际是延时71us。思路根本不对,读时隙至少延时60us,这里又误撞对了,把至少延时和读时隙间隔至少1us都包含进了。

}

return(dat); //返回读出的十进制数据

}

/*****************************************************

函数功能:向DS18B20写入一个字节数据

入口参数:dat

***************************************************/

WriteOneChar(unsigned char dat)

{

unsignedchar i=0;

for(i=0; i<8;>

{

DQ=1; // 先将数据线拉高

_nop_(); //等待一个机器周期

DQ=0; //将数据线从高拉低时即启动写时序

DQ=dat&0x01; //利用与运算取出要写的某位二进制数据,并将其送到数据线上等待DS18B20采样

for(time=0;time<10;time++);>延时约30usDS18B20在拉低后的约15~60us期间从数据线上采样。实际是延时87us

DQ=1; //释放数据线

for(time=0;time<1;time++);>延时3us,两个写时序间至少需要1us的恢复期 实际是延时15us

dat>>=1; //dat中的各二进制位数据右移1

}

for(time=0;time<4;time++);>稍作延时,给硬件一点反应时间。延时39us

}

/*****************************************************

函数功能:做好读温度的准备

***************************************************/

void ReadyReadTemp(void)

{

Init_DS18B20(); //DS18B20初始化

WriteOneChar(0xCC); // 跳过读序号列号的操作

WriteOneChar(0x44); // 启动温度转换

for(time=0;time<100;time++);>温度转换需要一点时间。延时807us

Init_DS18B20(); //DS18B20初始化

WriteOneChar(0xCC);//跳过读序号列号的操作

WriteOneChar(0xBE);//读取温度寄存器,前两个分别是温度的低位和高位

}

/*****************************************************

函数功能:延时若干毫秒

入口参数:n

***************************************************/

voiddelaynms(unsigned char n)

{

unsigned char i;

for(i=0;i<>

delay1ms();

}

照理说void delaynms(unsigned char n),参数n的取值范围是0~255,因是是无符号数。但在主函数中却调用delaynms(1000),明显错误,超出取值范围。



  • C51代码优化


2楼回复:
程序的注释,不是完全对。
程序通过返复调整得到正确代码,这是每一个程序员编程过程,
在这过程中不可能总是修改注释,
这样就导致注释的不同步。
延时函数一般情况下是线性。
原程序的运行机器周期可能不是1us

楼主回复:
楼上,我敢保证我的想法是对的。因为把源程序原封不动写入芯片,测温是正常显示的,晶振是12Mhz。当然引脚也是对应的。

这要动脑再想想哪程序错了,为什么还能正常用,的确运气很好。例如源程序的另一个延时程序又是错的,其注释思路肯定不对,但其实际效果还是接近1ms的:
/*****************************************************
函数功能:延时1ms
(3j+2)*i=(3×33+2)×10=1010(微秒),可以认为是1毫秒
***************************************************/
void delay1ms()
{
unsigned char i,j;
for(i=0;i<>
for(j=0;j<33;j++);>
}

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多