首页 > 编程语言 >简易数字电压表+ADC0809+程序查询(延时等待)方式

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

时间:2023-02-02 21:56:13浏览次数:63  
标签:__ LedBuf 定时器 void ADC0809 延时 电压表 GPIO include

1 实验现象

2 实验原理

3 系统设计

4 硬件设计

5 软件设计

5.1 主程序

#include <REG52.H>
#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 <REG52.H>

#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 <reg52.h>
//#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 <reg52.h>

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 <reg52.h>

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

标签:__,LedBuf,定时器,void,ADC0809,延时,电压表,GPIO,include
From: https://www.cnblogs.com/zclv/p/17087531.html

相关文章

  • 赛灵斯zynq standalone 延时函数
    voiddelay_ms(unsignedintmseconds){XTimetEnd,tCur;XTime_GetTime(&tCur);tEnd=tCur+(((XTime)mseconds)*(COUNTS_PER_SECOND/1000));do{XTim......
  • rabbitmq 延时消息队列
    //rabbitmq延时消息队列生产端demo//1.将消息发送到延时交换机对应的队列上delay-queue,指定过期时间;过期后转发的交换机和绑定的key//2.过期时间过期......
  • 只需两步便可生成 51 单片机最精准的延时函数
    前言我们在学习51单片机的过程中会用到延时,比如一个简单的流水灯就需要延时来控制依次点亮的时间,或者一些模块在单片机发出读数据指令后,需要延时几十微秒才可以读出数据......
  • RabbitMQ延时队列
    1.延时队列定时关单RabbitMQ:7.延迟队列https://www.cnblogs.com/yydscn/p/15208402.html......
  • 以修改注册表的方式避免ACK确认机制带来的延时现象
    TCP本身属面向链接的通讯协议。通讯双方的每一个收发动作,需要以通讯链路正常为前提。因此TCP协议内部提供了默认的ACK验证机制。假定A、B之间存在一条TCP通讯链路,某一时刻A......
  • netcore下RabbitMQ队列、死信队列、延时队列及小应用
    关于安装rabbitmq这里一笔掠过了。下面进入正题:1.新建aspnetcorewebapi空项目,NormalQueue,删除controllers文件夹已经无关的文件,这里为了偷懒不用console控制台:public......
  • 轻松解决企业数据高延时,华为云全球加速GA使用更灵活!
    轻松解决企业数据高延时,华为云全球加速GA使用更灵活!对于业务庞大且分布地区较为分散的企业来说,常常会出现访问总部的公司数据延时高、速率低等问题,从而降低了工作效率,增加了......
  • 延时方案
    数据库轮循原理:定时间隔一定时间查询数据库,获取即将到期的任务执行缺点:时效性差(间隔指定时间查询,难以做到准时)、任务量大会有风险(比如某次数据量过大,可能这个间隔时间都......
  • QT实现延时
    今天写程序需要QT代码用到延时__ms的功能//延时10msQEventLooploop;QTimer::singleShot(10,&loop,SLOT(quit()));loop.exec();参考:QT不阻塞线程,达到延时的作用_......
  • 华为云桌面,高流畅低延时,给你丝滑办公体验​
    华为云桌面,高流畅低延时,给你丝滑办公体验​随着数字化转型浪潮的到来,企业的办公方式也在发生变革,云办公正在被众多企业选择。不管是远程办公、远程视频会议、在线教育还是在......