模数转换ADC0804的应用(含源程序及电路)
[实验要求]
从ADC0804 的模拟量通道输入0-5V 之间的模拟量,通过ADC0804 转换成数字量送给单片机,经单片机处理后在数码管上以十进制形成显示出来。
[实验目的]
学习如果用单片机控制ADC0804芯片进行数模转换,掌握数码管动态扫描显示的原理。
动态扫描:就六位数码管显示123456举例说明如下:先让第一个数码管显示1,其余的全部不亮,1大约亮几毫秒,然后熄灭,紧接着立即让第二个数码管显示2,其余的全部不亮,2同样亮几毫秒,依次这样亮到第六个数码管,然后再回来显示1,如此这样以很快的速度不断循环下去,由于人眼的视觉暂留时间大约为20毫秒左右,所以是感觉不出有不亮的数码管存在的,看见的是六个数码管同时在显示,数值是123456,如果我们把这个过程一点点放慢,看见的是从第一个数码管显1,然后移到第二个再显2,。。。也就是说在任一时刻只有一位数码管是亮的。这就是数码管动态扫描显示的原理
ADC0804: ADC0804是8位全MOS中速A/D 转换器、它是逐次逼近式A/D 转换器,片内有三态数据输出锁存器,可以和单片机直接接口。单通道输入,转换时间大约为100us。ADC0804 转换时序是:当CS=0 许可进行A/D 转换。WR由低到高时,A/D开始转换,一次转换一共需要66-73 个时钟周期。CS与WR同时有效时启动A/D转换,转换结束产生INTR 信号(低电平有效),可供查询或者中断信号。在CS和RD 的控制下
可以读取数据结果。本实验没有使用INTR信号。
[硬件电路]
[源代码]
//拧动AD旁边的电位器,会在数码管的前三位显示0-255之间的数//值。这就是把模拟信号转换成数字信号,即模数转换。
#include #include #define uint unsigned int #define uchar unsigned char sbit adrd=P3^7; //IO口定义 sbit adwr=P3^6; sbit diola=P2^5; sbit dula=P2^6; sbit wela=P2^7; unsigned char j,k,adval; void delay(unsigned char i) //延时程序 { for(j=i;j>0;j--) for(k=125;k>0;k--); } uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d, //数码管编码 0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; void display(uchar bai_c,uchar sh_c,uchar g_c) //显示程序 { P0=table[bai_c]; //显示百位 dula=1; dula=0; P0=0xfe; wela=1; wela=0; delay(5); dula=0; P0=table[sh_c]; //显示十位 dula=1; dula=0; wela=0; P0=0xfd; wela=1; wela=0; delay(5); P0=table[g_c]; //显示个位 dula=1; dula=0; P0=0xfb; wela=1; wela=0; delay(5); } void main() // 主程序 { uchar a,A1,A2,A2t,A3; while(1) { wela=1; P0=0; //选通ADCS adwr=0; //AD写入(随便写个什么都行,主要是为了启动AD转换) _nop_(); adwr=1; P0=0xff; //关闭ADCS delay(10); wela=0; //关闭有AD片选信号锁存器的锁存端以防止在操作数码管时使AD的片选发生变化 for(a=20;a>0;a--) //需要注意的是ADC0804在写和读之间的时间间隔要足够长否则 无法读出数据 { //这里把显示部分放这里的原因也是为了增加写读之间的时间隔 display(A1,A2,A3); } //送去显示各位。 wela=1; //重新打开有AD片选信号锁存器的锁存端 P1=0xff; //读取P1口之前先给其写全 P0=0; //选通ADCS adrd=0; //AD读使能 adval=P1; //AD数据读取赋给P1口 adrd=1; P0=0xff; //关闭ADCS adwr=0; P1=adval; //同时把AD的值送八个发光二极显示 A1=adval/100; //分出百,十,和个位 A2t=adval%100; A2=A2t/10; A3=A2t%10; }; } 因篇幅问题不能全部显示,请点此查看更多更全内容