上一篇我们已经讲了ADC0809的原理与简化电路,仔细检查电路的正确性,尤其不能有短路现象。现在我们将它实际接上S52板子,静态电流为4mA,其中LED用了1mA,也就是说ADC0809静态电流为3mA左右。

将ADC0809板插上S52板,数据口D0-D7我接的是P0,CLK接P1.0,OE、EOC、ST-ALE分别接P1.1、P1.2、P1.3,Vcc和GND接S52板上的电源和地。
下面讲程序:
程序分为三个部分:主程序、LCD显示、ADC0809转换。

左边的项目框里有三个程序文件:主程序ADC0809m.C、LCD显示程序12864put.c、ADC0809转换程序ADC0809C.asm。嗯?。。。最后一个文件是汇编文件?是的!KEIL允许同时加入C程序和汇编程序一块儿编译。我们先来看主程序:
#include <AT89X52.H>
#define uchar unsigned char
extern void LcmClear( void ); //清屏,外部函数
extern void LcmInit( void ); //初始化,外部函数
extern void LcmPutstr( uchar row,uchar y,uchar * str ); //在设定位置显示字符串
extern uchar adc0809conv(void); //
uchar * uchartostr(unsigned char unm); //将char值转成字符串
uchar str[4]; //定义四个字节的数组,用来存放将数值转成的字符
//****************************
//将char值转成字符串函数
//****************************
uchar * uchartostr(uchar unm)
{
uchar x00,xx,x0,x,n; //定义百位,十位,个位变量
x00=unm/100;
xx=unm%100;
x0=xx/10;
x=xx%10;
n=0;
if(x00!=0)
{ str[n]=x00+48; //值加48即为字符
n++;
}
if(!(x00==0&x0==0))
{ str[n]=x0+48;
n++;
}
str[n]=x+48;
n++;
str[n]='\0';
return str;
}
//****************
// 主函数
//****************
void Main( void )
{ uchar aa; //定义一个临时字符变量
/* T2 set */
TR2=0x0; //停止T2定时器
TR0=0x0; //停止T0定时器
T2MOD=0x02; //0010(B) 设置T2为P1.0口输出方波模式
C_T2=0; //用内部时钟计数
TL2=0xfd;
TH2=0xff;
RCAP2L=0xfd;
RCAP2H=0xff;
TMOD=0x01; //设置T0为1定时模式(16位计数)
TH0=0;
TL0=0;
TR2=1; //打开T2定时器,开始输出脉冲
aa=adc0809conv(); //启动一次ADC0809转换并将值交给aa
LcmInit(); //初始化LCD
LcmClear(); //LCD清屏
LcmPutstr( 2,28,"ADC0809 TEST" );
LcmPutstr( 4,59,uchartostr(aa) ); //在第四行第59列输出ADC0809转换的值
LcmPutstr( 7,42,"TXZ001@" );
while(1)
{
}
}
下面我们再来看ADC0809转换函数:
NAME ADC0809C
?PR?adc0809conv?ADC0809C SEGMENT CODE
?DT?adc0809conv?ADC0809C SEGMENT DATA OVERLAYABLE
PUBLIC adc0809conv
RSEG ?DT?adc0809conv?ADC0809C
?adc0809conv?BYTE:
put?040: DS 1
RSEG ?PR?adc0809conv?ADC0809C
adc0809conv: ;程序从这里开始
USING 0
st bit P1.3 ;设置ST接P1.3
eoc bit P1.2 ;设置EOC接P1.2
oe bit P1.1 ;设置OE接P1.1
port equ P0 ;设置数据读取PORT接P0
setb TR0 ;启动T0定时器,用来计数(我是用T0来计算转换一次需要多长时间)
clr oe ;初始化ADC0809,OE置0
clr st ;初始化ADC0809,ST置0
setb eoc ;初始化ADC0809,EOC置1
mov port,#0 ;先择通道0数据交给P0口
setb st ;这三句将ST给出一个正脉冲来启动转换
nop ;
clr st ;
mov r7,#10 ;这两行是用来稍做延时
djnz r7,$ ;
wait1: jb eoc,wait1 ;这两行是来检测EOC由低到高发出了上升沿,以表示转换结束
wait2: jnb eoc,wait2 ;
mov port,#0FFh ;将P0口复位,以便下一步读取数据
setb oe ;将OE口置1,允许转换后的数据读出
clr TR0 ;停止T0计时器(T0是从0开始计数的,到这儿转换结束停止计数)
MOV R7,port ;将转换的数据交给主调用程序的变量aa
?C0001:
RET ;返回
END
汇编程序有点乱,没关系,下一篇我会专门讲混合编程。
那个LCD显示函数就不在这儿列出了,前面都已讲过也列出了程序。
下图是本程序实际测量一节AA电池的实例图:

可以看到,它的值为64,因为我的ADC0809参考电压为5V,那么8位精度的转换是将5v分为255份那么每份就是5÷255=0.0196v。我测量出一节电池的64值就为64×0.0196=1.25v。
调试注意点:由于8个模拟测量通道的输入阻抗很高,在程序运行时如果8个模拟端是悬空的,模拟端的电位是随周围环境变化的,那测量出的很可能是乱跳的随机值而并非你程序或电路问题。要避免这种情况的干扰,最好先用10K的电阻将模拟端接地。等你测量出每次都为0时,再改变模拟端的电位试验测量的正确性。
随便说说转换速度的问题。
ADC0809的转换速度跟脉冲频率有关,它的允许范围为10KHZ--1.28M,我们是用T2定时器来做脉冲输出的,频率就由公式 晶振频率/(4×(65536-(RCAP2H,RCAP2L)),还记得上一篇我给出过的这个公式吗?我的晶振是12MHZ,那么要给出1MHZ的脉冲就要在RCAP2里给65533的值,这样12M÷4×3=1MHZ。同理,要输出10KHZ的脉冲就要给65236的值。在主程序里:
TL2=0xfd ;这是计数器里的初值,FFFD就是65533,也就是输出1MHZ的脉冲
TH2=0xff ;
RCAP2L=0xfd ;这是重载器,也一样给上65533的值。
RCAP2H=0xff ;
如果要想输出10KHZ的脉冲上面就要给上65236的值,也就是FED4。
在主程序里我还用了T0计数器:
TMOD=0x01; //设置T0为1定时模式(16位计数)
TH0=0;
TL0=0;
我给的初值为0,但我没有在主程序里启动它。而是在ADC0809转换函数里才启动和停止:
setb TR0 ;启动T0定时器,用来计数(我是用T0来计算转换一次需要多长时间)
。。。。
clr TR0 ;停止T0计时器(T0是从0开始计数的,到这儿转换结束停止计数)
这样我就可以看它计数的值来知道一次转换需要多长时间了。将转换函数的最后一句:
MOV R7,port 换成 MOV R7,TL0 或 MOV R7,TH0 就会在LCD上显示出转换所用的时间了,因为晶振为12MHZ,一个脉冲就是一微秒。实际测试ADC0809在1MHZ时钟时转换一次为83微秒,而在10KHZ时钟下转换一次需要1769微秒也就是1.769毫秒,比ADC0804转换速度要慢很多,ADC0804为22微秒。
怎么样,单片机好玩吧!