单片机
返回首页

简易数字电压表+ADC0809+程序查询(延时等待)方式

2024-07-26 来源:cnblogs

1 实验现象

2 实验原理

3 系统设计

4 硬件设计

5 软件设计


5.1 主程序


#include

#include 'DisplaySmg.h'

#include 'ADC0809.h'

#include 'Timer0.h'

#include 'Timer1.h'


unsigned char  adc_result = 0;

int adc_result_show = 0;


void disp_num(void)            //显示四位十进制数

{

    LedBuf[0]= 23;                    //千位,不显示

    LedBuf[1]= adc_result_show/100;    //百位

    LedBuf[2]= adc_result_show/10%10;//十位    

    LedBuf[3]= adc_result_show%10;    //个位

}


void main()

{

    Timer0_Init();        //定时/计数器T0初始化

    Timer1_Init();        //定时/计数器T1初始化

    EA=1;                //中断总开关

    DotDig1=1;            //点亮第二个数码管的小数点

    while(1)

    {

        adc_result = ADC_Conv();

        adc_result_show = adc_result*1.0*100*5/255;    //数据变换处理(线性标度变换)

        disp_num();        //显示数据

    }

}


void Timer0_ISR(void) interrupt 1 

{

    TR0=0;                //关闭定时器

    DisplaySmg();        //每过1ms,刷新一次数码管显示函数

    TL0 = 0x66;            //设置定时初始值,定时1ms

    TH0 = 0xFC;            //设置定时初始值,定时1ms

    TR0=1;                //打开定时器

}


5.2 数码管显示模块


#ifndef __DisplaySmg_H__

#define __DisplaySmg_H__


#include


#define GPIO_SEG P0        //段选端

#define GPIO_SEL P2        //位选端


extern unsigned char LedBuf[];    //外部变量声明

extern unsigned char DotDig0,DotDig1,DotDig2,DotDig3;


void DisplaySmg(void);


#endif


#include 'DisplaySmg.h'


unsigned char code LedData[]={    //共阴型数码管的段码表,字符,序号

                0x3F,  //'0',0

                0x06,  //'1',1

                0x5B,  //'2',2

                0x4F,  //'3',3

                0x66,  //'4',4

                0x6D,  //'5',5

                0x7D,  //'6',6

                0x07,  //'7',7

                0x7F,  //'8',8

                0x6F,  //'9',9

                0x77,  //'A',10

                0x7C,  //'B',11

                0x39,  //'C',12

                0x5E,  //'D',13

                0x79,  //'E',14

                0x71,  //'F',15

                0x76,  //'H',16

                0x38,  //'L',17

                0x37,  //'n',18

                0x3E,  //'u',19

                0x73,  //'P',20

                0x5C,  //'o',21

                0x40,  //'-',22

                0x00,  //熄灭 23

                         };

unsigned char DotDig0=0,DotDig1=0,DotDig2=0,DotDig3=0;    //小数点控制位

unsigned char code LedAddr[]={0xfe,0xfd,0xfb,0xf7};        //数码管位选

unsigned char LedBuf[]={22,22,22,22};    //显示缓存区


void DisplaySmg()                    //四位数码管,考虑小数点

{

    unsigned char     i;                 //等价于 'static unsigned char i = 0;'

    unsigned char     temp;

    switch(i)

    {

        case 0:

        {

            GPIO_SEG = 0x00;                //消影

            if(DotDig0==1)                    //小数点

            {

                temp = LedData[LedBuf[0]] | 0x80;  //点亮小数点

            }

            else

            {

                temp = LedData[LedBuf[0]];            

            }

            GPIO_SEG = temp;                //段码

            GPIO_SEL = LedAddr[0];            //位选

            i++;

            break;

        }

            

        case 1:

            GPIO_SEG = 0x00;    

            if(DotDig1==1)                    //小数点

            {

                temp = LedData[LedBuf[1]] | 0x80;

            }

            else

            {

                temp = LedData[LedBuf[1]];

            }

            GPIO_SEG = temp;

            GPIO_SEL = LedAddr[1];

            i++;

            break;

        case 2:

            GPIO_SEG = 0x00;

            if(DotDig2==1)                    //小数点

            {

                temp = LedData[LedBuf[2]] | 0x80;

            }

            else

            {

                temp = LedData[LedBuf[2]];

            }

            GPIO_SEG = temp;

            GPIO_SEL = LedAddr[2];

            i++;

            break;

        case 3:

            GPIO_SEG = 0x00;

            if(DotDig3==1)                    //小数点

            {

                temp = LedData[LedBuf[3]] | 0x80;

            }

            else

            {

                temp = LedData[LedBuf[3]];

            }

            GPIO_SEG = temp;

            GPIO_SEL = LedAddr[3];

            i=0;

            break;

        default:break;

    }

}


5.3 ADC0809模数转换模块


#ifndef __ADC0809_H__

#define __ADC0809_H__


#include

//#include 'DelayXms.h'


#define ADC_DATA P3


sbit ADDR_A = P1^0;

sbit ADDR_B = P1^1;

sbit ADDR_C = P1^2;

sbit CLK    = P1^3;

sbit START    = P1^4;

sbit EOC    = P1^5;

sbit OE        = P1^6;


unsigned char ADC_Conv(void);


#endif


#include 'ADC0809.h'


/*ADC0809的工作过程:

    首先输入3位地址,并使ALE=1,将地址输入地址锁存器中。此地址经

译码选通8路模拟输入之一到比较器。START上升沿将逐次逼近寄存器复位。

下降沿启动ADC转换,之后EOC输出信号变低,指示转换正在进行。直到ADC

转换完成,EOC变为高电平,指示ADC转换结束,结果数据已存入锁存器,

这个信号可用作中断申请。当OE输入高电平时,输出三态门打开,转换结果的

数字量输出到数据总线上。*/


unsigned char ADC_Conv()

{

    unsigned char adc_result;

    

    OE = 0;     //数据输出允许信号,高电平有效

    START = 0;     //ADC转换启动信号,高电平有效, 电路中与ALE(地址锁存允许信号)连在一起

    

    ADDR_A = 1;    //3位地址输入线

    ADDR_B = 1;    //用于选通8路模拟输入中的一路

    ADDR_C = 0;

    

    START = 1;    //上升沿,同时将ADC内部的寄存器清零

    START = 0;    //产生一定的脉冲,Typ=100ns,下降沿启动AD转换

    

    while(EOC==0); //查询方式,ADC转换结束信号,EOC=1,转换结束

//    DelayXms(1);    //延时等待方式

    OE = 1;        //数据更新到输出端口

    adc_result = ADC_DATA;    //单片机读取数据

    OE = 0;        //本次过程结束

    return adc_result;

}


//定时器T1产生500KHz时钟信号,即2us, 定时器每1us中断一次

void Timer1_ISR(void) interrupt 3 

{

    CLK =!CLK;

}


5.4 定时器T0


#ifndef __Timer0_H__

#define __Timer0_H__


#include


void Timer0_Init(void);


#endif


#include 'Timer0.h'


void Timer0_Init(void)        //1毫秒@11.0592MHz

{

    TMOD &= 0xF0;        //设置定时器模式

    TMOD |= 0x01;        //设置定时器模式

    TL0 = 0x66;            //设置定时初始值

    TH0 = 0xFC;            //设置定时初始值

    TF0 = 0;            //清除TF0标志

    TR0 = 1;            //定时器0开始计时

    

    ET0 = 1;            //定时器0中断开关

//    EA  = 1;             //中断总开关

}



//中断服务函数一定是一个没有返回值的函数

//中断服务函数一定是没有参数的函数

//中断服务函数函数名后跟着关键字interrupt

//interrupt n 0~4 5个中断源,8*n+0003H

// 0003H INT0, 00BH T0, 0013H INT1, 001BH T1, 0023H ES

//中断服务函数不能被主程序或者其他程序所调用

//n后面跟着using m(0~3)工作寄存器组


//void Timer0_ISR(void) interrupt 1 

//{

//    TL0 = 0x66;            //设置定时初始值

//    TH0 = 0xFC;            //设置定时初始值

//}


5.5 定时器T1


#ifndef __Timer1_H__

#define __Timer1_H__


#include


void Timer1_Init(void);


#endif


#include 'Timer1.h'


void Timer1_Init(void)        //1微秒@11.0592MHz

{

    //定时器时钟12T模式

    TMOD &= 0x0F;        //设置定时器模式

    TMOD |= 0x20;        //设置定时器模式,,8位自动重载模式

    TL1 = 0xFF;        //设置定时初始值,定时1us

    TH1 = 0xFF;        //设置定时重载值,定时1us

    TF1 = 0;        //清除TF1标志

    TR1 = 1;        //定时器1开始计时

    

    ET1 = 1;        //定时器1中断允许

}


6 参考来源


(1)单片机应用——ADC0809查询方式实现一路模拟信号的A/D转换(数码管显示)_哔哩哔哩_bilibili;


进入单片机查看更多内容>>
相关视频
  • RISC-V嵌入式系统开发

  • SOC系统级芯片设计实验

  • 云龙51单片机实训视频教程(王云,字幕版)

  • 2022 Digi-Key KOL 系列: 你见过1GHz主频的单片机吗?Teensy 4.1开发板介绍

  • TI 新一代 C2000™ 微控制器:全方位助力伺服及马达驱动应用

  • MSP430电容触摸技术 - 防水Demo演示

精选电路图
  • 红外线探测报警器

  • 短波AM发射器电路设计图

  • 使用ESP8266从NTP服务器获取时间并在OLED显示器上显示

  • 用NE555制作定时器

  • 带有短路保护系统的5V直流稳压电源电路图

  • 基于TDA2003的简单低功耗汽车立体声放大器电路

    相关电子头条文章