新概念51单片机C语言教程纠错(3)

这一次错误在课本P140页例6.6.1中,这个例子在proteus仿真中再一次刷新了我的认知。

主要的bug在两个地方,第一仍是单片机上电后数码管的点亮问题,由于这个例题需要数码管不断刷新,所以bug的影响比前一篇纠错文章中更加明显;第二是由于没加延迟函数导致的无脑刷新问题。

具体效果视频中有表现,其中的操作为,上位机输入1,单片机的ad打开,并返回voltage到调试助手,同时数码管显示数字。输入2,关闭ad。输入其他数字显示error。

CSDN视频2

解决方法:
bug1:按照上一篇文章的方法更改,文章链接:https://blog.csdn.net/toyjis/article/details/104096180,代码我写在文后。

bug2:郭在例题中的要求为,一秒钟数据刷新一次,但是仔细分析代码就会发现,课本例题代码中根本没有延迟一秒钟语句,所以我们加上即可,但是加的位置却很有讲究,既不能影响中断,也不能影响串口,所以我加到了printf后面(主要是为了方便执行)。
PS:一秒的延迟不利于数码管数字的连续性,所以建议改成500ms或者其他数据。

以下代码是书中例题,仍是错误的,只需要把第一段替换,第二段补充即可。



#include
#include
#include
#define uchar unsigned char
#define uint unsigned int
sbit dula=P2^6;
sbit wela=P2^7;
sbit adwr=P3^6;
sbit adrd=P3^7;
uchar flag,a;
unsigned char flag_uart,flag_time,flag_on,a,i,t0_num,ad_val;
float ad_vo;
uchar code table[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
void delayms(uint xms)
{
uint i,j;
for(i=xms;i>0;i--)
for(j=110;j>0;j--);
}
void xvnicuankou_init()
{
TMOD=0x20;
SCON=0x50;
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
TH1=0xfd;
TL1=0xfd;
TR1=1;
ET0=1;
SM0=0;
SM1=1;
REN=1;
EA=1;
ES=1;
}
void display(uchar value)
{
uchar bai,shi,ge;
bai=value/100;
shi=value%100/10;
ge=value%10;
dula=1;
P0=table[bai];
dula=0;
P0=0xff;
wela=1;
P0=0x7e;
wela=0;
delayms(5);
dula=1;
P0=table[shi];
dula=0;
P0=0xff;
wela=1;
P0=0x7d;
wela=0;
delayms(5);
dula=1;
P0=table[ge];
dula=0;
P0=0xff;
wela=1;
P0=0x7b;
wela=0;
delayms(5);
/*uchar bai,shi,ge;
bai=value/100;
shi=value%100/10;
ge=value%10;
P0=0xff;
wela=1;
P0=0x7e;
wela=0;
P0=0xff;
dula=1;
P0=table[bai];
dula=0;
delayms(5);
P0=0xff;
wela=1;
P0=0x7d;
wela=0;
P0=0xff;
dula=1;
P0=table[shi];
dula=0;
P0=0xff;
delayms(5);
P0=0xff;
wela=1;
P0=0x7b;
wela=0;
P0=0xff;
dula=1;
P0=table[ge];
dula=0;
P0=0xff;
delayms(5);*/      //此段为替换代码
}
uchar get_ad()
{
uchar adval;
adwr=1;
_nop_();
adwr=0;
_nop_();
adwr=1;
P1=0xff;
adrd=1;
_nop_();
adrd=0;
_nop_();
adval=P1;
adrd=1;
return adval;
}
void main()
{
xvnicuankou_init();
wela=1;
P0=0x7f;
wela=0;
while(1)
{
if(flag_uart==1)
{
flag_uart=0;
ES=0;
TI=1;
switch(flag_on)
{
case 0:puts("'turn on ad!!\n");
TR0=1;
break;
case 1:puts("turn off ad!!\n");
TR0=0;
break;
case 2:puts("error!!\n");
break;
}
while(!TI);
TI=0;
ES=1;
}
if(flag_time==1)
{
flag_time=0;
ad_val=get_ad();
ad_vo=(float)ad_val*5.0/256.0;
ES=0;
TI=1;
printf("THE voltage is %fV\n",ad_vo);
/*delayms(1000);*/            //此段为添加代码
while(!TI);
TI=0;
ES=1;
}
display(ad_val);
}
}
void timer0() interrupt 1
{
TH0=(65536-50000)/256;
TL0=(65536-50000)%256;
t0_num++;
if(t0_num==20)
{
t0_num=0;
flag_time=1;
}
}
void ser() interrupt 4
{
RI=0;
a=SBUF;
flag_uart=1;
if(a==1)
flag_on=0;
else if(a==2)
flag_on=1;
else
flag_on=2;
}

你可能感兴趣的:(新概念51单片机C语言教程纠错,单片机)