首页 > 其他分享 >简易数字电压表+ADC0832+串行SPI方式实现1路数据转换

简易数字电压表+ADC0832+串行SPI方式实现1路数据转换

时间:2023-02-09 17:44:53浏览次数:68  
标签:void unsigned char SPI adc 电压表 ADC0832 result

1 实物与模型

(1)为什么DO和DI引脚连在一个引脚上?

由于ADC0832在通信时并不是会同时使用DO和DI端口,并且DO和DI端口与单片机的接口是双向的,所以在设计电路中可以用一根线将DO端和DI端连接到一起。

(2)ADC0832的外部连接采用SPI总线结构,这样便把它的连接方式与其他设备统一起来了。ADC0832采用同步串行传输,同步是通过时钟线进行数据同步;串行传输通过DO 数据口一位一位传输数据。

 2 实验原理

3 系统设计

4 软件设计

1 ADC0832模数转换函数(结合ADC时序图)

#ifndef __ADC0832_H__
#define __ADC0832_H__

#include <reg52.h>
#include <intrins.h>

sbit ADC0832_CS_N = P1^0;
sbit ADC0832_CLK  = P1^1;
sbit ADC0832_DI   = P1^2;
sbit ADC0832_DO   = P1^2;

void ADC0832_Init(void);
unsigned char ADC0832_Conv(void);

#endif
#include "ADC0832.h"

//ADC0832初始化
void ADC0832_Init(void)
{
    ADC0832_CS_N = 1;
    ADC0832_CLK  = 0;
    ADC0832_DI   = 1;
}

unsigned char ADC0832_Conv(void)
{
    unsigned char adc_result1 = 0;        //用来接收第一组数据
    unsigned char adc_result2 = 0;        //用来接收第二组数据
    
    unsigned char i;
    
    //时序1,先将CS使能端置于低电平并且保持低电平直到转换完全结束
    ADC0832_CS_N = 0;    
    ADC0832_CLK  = 0;
    
    //时序2,由MCU向ADC0832时钟输入端CLK输入时钟脉冲,DO/DI端使用DI端输入通道功能选择的数据信号
    
    //在第1个时钟脉冲的下降沿之前DI端必须是高电平,表示启动信号
    ADC0832_DI   = 1;    //START BIT,启动信号
    _nop_();
    ADC0832_CLK  = 1;    //第一个脉冲
    _nop_();            
    ADC0832_CLK  = 0;
    
    //在第2、3个脉冲下降沿之前DI应输入2位数据用于选择通道功能
    ADC0832_DI   = 1;    //单通道输入
    _nop_();
    ADC0832_CLK  = 1;    //第二个脉冲
    _nop_();            
    ADC0832_CLK  = 0;
    
    ADC0832_DI   = 0;    //选择CH0作为模拟信号输入端
    _nop_();
    ADC0832_CLK  = 1;    //第三个脉冲
    _nop_();            
    ADC0832_CLK  = 0;
    
    /*时序3,从第4个脉冲开始由DO输出转换数据最高位,
    随后每个脉冲下降沿DO输出下一位数据,
    直到第11个脉冲时发出最低位数据,一个字节的数据输出完成*/
    
    ADC0832_DI   = 1;    //数据线拉高,主机准备读数据,高位在前
    for(i=0;i<8;i++)
    {
        ADC0832_CLK  = 1;
        _nop_();
        ADC0832_CLK  = 0;    //CLK下降沿
        adc_result1 = adc_result1 << 1;      //左移,0000_0001
        if(ADC0832_DO==1)
            adc_result1 = adc_result1 | 0x01;
    }
    
    /*时序4,随后输出8个位数,与前面数据顺序相反,
    同时第11个脉冲的下降沿输出DATA0,到第19个脉冲时输出完成DATA7*/
    for(i=0;i<8;i++)
    {
        adc_result2 = adc_result2 >> 1;      //右移,1000_000
        if(ADC0832_DO==1)
            adc_result2 = adc_result2 | 0x80;
        ADC0832_CLK  = 1;
        _nop_();
        ADC0832_CLK  = 0;    
    }
    
    //时序5 一次AD转换结束
    ADC0832_CS_N = 1;    
    ADC0832_CLK  = 1;
    ADC0832_DI   = 0;
    
    return (adc_result1 == adc_result2)? adc_result1:0;
}

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;
    }
}

3 定时器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;            //设置定时初始值
//}

4 主函数(对采集的数据进行均值滤波)

#include <REG52.H>
#include "DisplaySmg.h"
#include "ADC0832.h"
#include "Timer0.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()
{
    int adc_result_reg;
    int adc_result_fliter;        //采用均值滤波
    unsigned char adc_cnt;        //采样次数
    
    Timer0_Init();        //定时/计数器T0初始化
    ADC0832_Init();
    EA=1;                //中断总开关
    DotDig1=1;            //点亮第二个数码管的小数点
    while(1)
    {
        adc_result = ADC0832_Conv();    //采集数据,ADC转换后结果
        adc_result_reg = adc_result*1.0*100*5/255;    //数据变换处理(线性标度变换)
        adc_result_fliter = adc_result_fliter +  adc_result_reg;    //数据累加
        adc_cnt++;
        if(adc_cnt > 7)
        {
            adc_result_show = adc_result_fliter >> 3; //等价于除于8,取平均值
            adc_cnt = 0;
            adc_result_fliter = 0;
        }
        disp_num();        //显示数据
    }
}

void Timer0_ISR(void) interrupt 1 
{
    TR0=0;                //关闭定时器
    DisplaySmg();        //每过1ms,刷新一次数码管显示函数
    TL0 = 0x66;            //设置定时初始值,定时1ms
    TH0 = 0xFC;            //设置定时初始值,定时1ms
    TR0=1;                //打开定时器
}

 5 参考来源

(1)(141条消息) 【mcuclub】模数转换ADC0832_单片机俱乐部--官方的博客-CSDN博客_adc0832模数转换原理

(2)单片机应用——利用串行A/D转换器件ADC0832实现模拟电压信号的A/D转换_哔哩哔哩_bilibili

标签:void,unsigned,char,SPI,adc,电压表,ADC0832,result
From: https://www.cnblogs.com/zclv/p/17104793.html

相关文章

  • SpirngBoot+SpringSecurity+Jwt鉴权认证
    写在开头基础环境jdk1.8+maven3.8.5。为了简单起见,省略从表中查询数据,直接从dao返回用户权限数据。引入依赖<!--springboot--><dependency>......
  • Spider实战系列-抓取《一人之下第三季》
    今天我们抓取的是m3u8的视频,视频有长视频和短视频之分.抓取m3u8类型视频对于短视频一般来说一个视频对应的就是一个url长视频  一个视频就几百兆到几十G不等 这种视频......
  • spi 介绍
    1,定义接口publicinterfaceLogger{voidinfo();}2,定义实现类publicclassLogbackimplementsLogger{@Overridepublicvoidinfo(){S......
  • 简易数字电压表+ADC0809+总线方式实现一路数据转换
    1实验现象2实验原理(略)3系统设计(略)4硬件设计5软件设计5.1主函数#include"DisplaySmg.h"#include"ADC0809.h"#include"Timer0.h"unsignedcharadc_r......
  • spi~在插件开发过程中的使用
    spi是原生java的组件,通过META-INF/services目录进行注册,通过ServiceLoader进行加载,一般可以用在组件开发中,你在公用组件中封装好逻辑,将个性化的部分抽象出一个接口,接口通过......
  • SPI-CH32V307VCT6实现SPI从机DMA收发
                SPI从机DMA收发 使用CH32V307VCT6单片机实现主从SPI数据交换    SPI从机IO口配置  从机SPI配置,注意使能SPI的DMA ......
  • Dubbo2.7的Dubbo SPI实现原理细节
    总结/朱季谦本文主要记录我对DubboSPI实现原理的理解,至于什么是SPI,我这里就不像其他博文一样详细地从概念再到JavaSPI细细分析了,直接开门见山来分享我对DubboSPI的见解......
  • QT中级(1)QTableView自定义委托(一)实现QSpinBox、QDoubleSpinBox委托
    1写在前面的话我们在之前写的《QT(7)-初识委托》文章末尾提到,“使用一个类继承QStyledItemDelegate实现常用的控件委托,在使用时可以直接调用接口,灵活实现各种委托”。我......
  • ESP_IDF中使用TFT_eSPI库驱动ST7789V
    前言:想学习创建好看的菜单界面很久了,寒假在家正好有时间,手中恰好有一块ST7789的tft屏幕,正好拿来练练手。Step1:在github中找到TFT_eSPI库(网址:docs·master......
  • 简易数字电压表+ADC0809+中断方式实现一路数据转换
    1实验现象2实验原理ADC0809的工作过程:首先输入3位地址,并使ALE=1,将地址输入地址锁存器中。此地址经译码选通8路模拟输入之一到比较器。START上升沿将逐次逼近寄存......