基于STC12C5A60S2系列1T 8051单片的模数芯片ADC0809实现模数转换应用

本文主要是介绍基于STC12C5A60S2系列1T 8051单片的模数芯片ADC0809实现模数转换应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

基于STC12C5A60S2系列1T 8051单片的模数芯片ADC0809实现模数转换应用

  • STC12C5A60S2系列1T 8051单片机管脚图
  • STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置
  • STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍
  • 模数芯片ADC0809介绍
    • 通过模数芯片ADC0809把电压模拟量转化为电压数字量

STC12C5A60S2系列1T 8051单片机管脚图

在这里插入图片描述在这里插入图片描述

STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置

在这里插入图片描述

STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍

在这里插入图片描述在这里插入图片描述

模数芯片ADC0809介绍

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

通过模数芯片ADC0809把电压模拟量转化为电压数字量

在这里插入图片描述在这里插入图片描述

#include <stc12c5a60s2.h>
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint 
#define NixieTubeSegmentCode P0//自定义数码管段码为单片机P0组引脚
#define NixieTubeBitCode P2//自定义数码管位码为单片机P2组引脚
#define ADC0809Data P3//自定义ADC0809数据变量为单片机P3组引脚
//#define KeyPressDeshakeTime 10//自定义按键按下消抖时间为10ms
//#define KeyLongPressDelayTime 500//自定义按键长按延时时间为500ms
//#define KeyLongPressIntervalChangeTime 25//自定义按键长按间隔变化时间为25ms
//uchar AddKeyLockFlag;//声明增加按键锁定标志位变量
//uchar DecKeyLockFlag;//声明减少按键锁定标志位变量
//uchar KeyNumber = 0;//定义按键键值为0
//uchar AddKeyLongPressAddIntervalTime;//声明增加按键长按连增间隔时间变量
//uchar DecKeyLongPressDecIntervalTime;//声明减少按键长按连减间隔时间变量
//uchar NumberValue;//声明数字量变量
//uint AddKeyPressDelayTime;//声明增加按键按下延时时间变量
//uint DecKeyPressDelayTime;//声明减少按键按下延时时间变量
uchar Code NixieTubeBitCodeArray = [0xfe,0xfd,0xfb,0xf7];//定义共阴数码管位码数组变量
uchar NixieTubeDisplayDataArray[0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,0x87,0xff,0xef,0xf7,0xfc,0xb9,0xde,0xf9,0xf1,0x40,0x00];//定义共阴数码管显示0~F、0~F带小数点数据及符号“—”及熄灭数组变量
uchar NixieTubeCacheDataArray[];//定义数码管缓存数据数组变量
uint OutPutVoltage;//声明输出电压变量
uint AnalogFilterOutPutVoltage;//声明模拟滤波后输出电压变量
//uint Timer0TimeCount;//声明定时器0定时计数变量
//sbit AddKey = P2^0;//位定义增加按键为单片机P2.0引脚
//sbit DecKey = P2^1;//位定义减少按键为单片机P2.1引脚
sbit ADC0809ADDRA = P1^0;//位定义ADC0809ADDRA模拟通道变量为P1.0端口
sbit ADC0809ADDRB = P1^1;//位定义ADC0809ADDRB模拟通道变量为P1.1端口
sbit ADC0809ADDRC = P1^2;//位定义ADC0809ADDRC模拟通道输入变量为P1.2端口
sbit ADC0809CLK = P1^3;//位定义ADC0809CLK时钟变量为P1.3端口
sbit ADC0809Star = P1^4;//位定义ADC0809Star启动变量为P1.4端口
sbit ADC0809EOC = P1^5;//位定义ADC0809EOC输出引脚变量为P1.5端口
sbit ADC0809OE = P1^6;//位定义ADC0809OE输出使能变量为P1.6端口void ADC0809Change()//ADC0809转化函数
{uchar AnalogChangeResult;//声明模拟转换结果变量ADC0809OE = 0;//ADC0809OE输出使能变量置低电平ADC0809Star = 0;//ADC0809Star启动变量置低电平ADC0809ADDRA = 1;//选ADC0809模拟通道3ADC0809ADDRB = 1;ADC0809ADDRC = 0;ADC0809Star = 1;//ADC0809Star启动变量置高电平 锁存ADC0809模拟通道3输入ADC0809Star = 0;//启动ADC0809模数转化while(ADC0809EOC == 0);//当ADC0809EOC输出引脚变量置低电平 表示ADC0809正在模数转化ADC0809OE = 1;//ADC0809OE输出使能变量置高电平AnalogChangeResult = ADC0809Data;//ADC0809数据变量包含的数据赋给AnalogChangeResult模拟转换结果变量ADC0809OE = 0;//ADC0809OE输出使能变量置低电平return AnalogChangeResult;//返回模拟转换结果变量包含的数据}/****void KeyScan()//按键扫描函数 该函数放在定时器定时1ms的中断函数中扫描
{if(AddKey)//如果增加按键没按下或弹起{AddKeyLockFlag = 0;//增加按键锁定标志位清0AddKeyPressDelayTime = 0;//增加按键按下延时时间清0}   else if(!AddKeyLockFlag)//如果增加按键锁定标志位置1 即增加按键按下{AddKeyPressDelayTime++;//增加按键按下延时时间自加if(AddKeyPressDelayTime > KeyPressDeshakeTime)//如果增加按键按下延时时间大于按键按下消抖时间{AddKeyPressDelayTime = 0;//增加按键按下延时时间清0KeyNumber = 1;//按键键值置1 此处是单击增加 可赋给swicth()语句中的变量来对数值单击增加AddKeyLockFlag = 1;//增加按键锁定标志位置1}}else if(AddKeyPressDelayTime < KeyLongPressDelayTime)//如果增加按键按下延时时间小于按键长按延时时间{AddKeyPressDelayTime++;//增加按键按下延时时间自加}else//如果增加按键按下延时时间大于按键长按延时时间{AddKeyLongPressAddIntervalTime++;//增加按键长按连增间隔时间自加if(AddKeyLongPressAddIntervalTime > KeyLongPressIntervalChangeTime)//如果增加按键长按连增间隔时间大于按键长按间隔变化时间{AddKeyLongPressAddIntervalTime = 0;//增加按键长按连增间隔时间清0KeyNumber = 1;//按键键值置1 此处是连击增加 可赋给swicth()语句中的变量来对数值连击增加}}   if(DecKey)//如果减少按键没按下或弹起{DecKeyLockFlag = 0;//减少按键锁定标志位清0DecKeyPressDelayTime = 0;//减少按键按下延时时间清0}   else if(!DecKeyLockFlag)//如果减少按键锁定标志位置1 即减少按键按下{DecKeyPressDelayTime++;//减少按键按下延时时间自加if(DecKeyPressDelayTime > KeyPressDeshakeTime)//如果减少按键按下延时时间大于按键按下消抖时间{DecKeyPressDelayTime = 0;//减少按键按下延时时间清0KeyNumber = 2;//按键键值置2 此处是单击减少 可赋给swicth()语句中的变量来对数值单击减少DecKeyLockFlag = 1;//减少按键锁定标志位置1}}else if(DecKeyPressDelayTime < KeyLongPressDelayTime)//如果减少按键按下延时时间小于按键长按延时时间{DecKeyPressDelayTime++;//减少按键按下延时时间自加}else//如果减少按键按下延时时间大于按键长按延时时间{DecKeyLongPressDecIntervalTime++;//减少按键长按连减间隔时间自加if(DecKeyLongPressDecIntervalTime > KeyLongPressIntervalChangeTime)//如果减少按键长按连减间隔时间大于按键长按间隔变化时间{DecKeyLongPressDecIntervalTime = 0;//减少按键长按连减间隔时间清0KeyNumber = 2;//按键键值置2 此处是连击减少 可赋给swicth()语句中的变量来对数值连击减少}}}****//****void NumberValueSet()//数字量数值设置函数
{switch(KeyNumber)//按键类型筛选位{case 1 ://增加按键单击、长按触发位NumberValue++;//数字量数值自加if(NumberValue > 255)//如果数字量数值大于255 为啥数字量数值变量NumberValue取255来比较?由于数字量数值变量NumberValue要计入DAC0832转换器 而DAC0832转换器是八位寄存器 最大只能计入255 因此数字量数值变量NumberValue取255来比较{NumberValue = 255;//数字量数值等于255}KeyNumber = 0;//按键键值清0break;//跳出case 2 ://减少按键单击、长按触发位NumberValue--;//数字量数值自减if(NumberValue < 0)//如果数字量数值小于0{NumberValue = 0;//数字量数值清0}KeyNumber = 0;//按键键值清0break;//跳出default:break;//跳出}}****/void NixieTubeDisplayDataSplit()//数码管显示数据分解函数
{uchar NixieTubeQianWei,NixieTubeBaiWei,NixieTubeShiWei,NixieTubeGewei;//声明数码管千位、百位、十位、个位变量NixieTubeQianWei = AnalogFilterOutPutVoltage / 1000 ;//数码管千位分解NixieTubeBaiWei = AnalogFilterOutPutVoltage / 100 % 10;//数码管百位分解NixieTubeShiWei = AnalogFilterOutPutVoltage / 10 % 10 ;//数码管十位分解NixieTubeGeWei = AnalogFilterOutPutVoltage % 10 ;//数码管个位分解NixieTubeCacheDataArray[0] = NixieTubeQianWei + 15;//数码管千位显示带小数点数据NixieTubeCacheDataArray[1] = NixieTubeBaiWei;//数码管百位显示数据NixieTubeCacheDataArray[2] = NixieTubeShiWei;//数码管十位显示数据NixieTubeCacheDataArray[3] = NixieTubeGeWei;//数码管个位显示数据}void NixieTubeDisplayData()//数码管显示数据函数  
{  static uchar i = 0;//定义静态数码管位变化变量switch(i)//数码管位变化筛选{case 0 ://数码管千位显示NixieTubeSegmentCode = 0x00;//数码管段码消影NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[0]];//数码管千位的段码显示NixieTubeBitCode = NixieTubeBitCodeArray[0];//数码管千位码显示i++;//数码管位变化自加1break;//跳出case 1 ://数码管百位显示NixieTubeSegmentCode = 0x00;//数码管段码消影NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[1]];//数码管百位的段码显示NixieTubeBitCode = NixieTubeBitCodeArray[1];//数码管百位码显示i++;//数码管位变化自加1break;//跳出 case 2 ://数码管十位显示NixieTubeSegmentCode = 0x00;//数码管段码消影NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[2]];//数码管十位的段码显示NixieTubeBitCode = NixieTubeBitCodeArray[2];//数码管十位码显示i++;//数码管位变化自加1break;//跳出case 3 ://数码管个位显示NixieTubeSegmentCode = 0x00;//数码管段码消影NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[3]];//数码管个位的段码显示NixieTubeBitCode = NixieTubeBitCodeArray[3];//数码管个位码显示i = 0;//数码管位变化清0break;//跳出default:break;//跳出}}  
/*****关于8051系列单片机定时器溢出率和定时器初值(定时计数初值)之间计算的知识点*****/ 
/****
一、定时器溢出率计算公式
1、定时器溢出率:定时器每秒溢出的次数
2、定时器溢出率计算公式表定时方式            分频方式                      公式
方式1:16位定时器  12分频(即12T 默认值)   Ft=晶振频率/12/(65536-定时器初值)
方式2:8位定时器   12分频(即12T 默认值)   Ft=晶振频率/12/(256-定时器初值)
方式1:16位定时器      1分频(即1T)       Ft=晶振频率/1/(65536-定时器初值)
方式2:8位定时器       1分频(即1T)       Ft=晶振频率/1/(256-定时器初值)
二、定时器初值(定时计数初值)计算公式定时方式            分频方式                      公式
方式1:16位定时器   12分频(即12T 默认值)  定时器初值(定时计数)=65536-晶振频率/12*定时时间
方式2:8位定时器    12分频(即12T 默认值)  定时器初值(定时计数)=256-晶振频率/12*定时时间
方式1:16位定时器       1分频(即1T)      定时器初值(定时计数)=65536-晶振频率*定时时间
方式2:8位定时器        1分频(即1T)      定时器初值(定时计数)=256-晶振频率*定时时间
****/void Timer0Init()//定时器0的16位定时模式1用12分频定时1ms初始化函数 晶振为12MHz
{AUXR &= 0x7f;//设定定时器/计数器模式为12TTMOD &= 0xf0;//设定定时器/计数器工作模式清0TMOD |= 0x01;//设定定时器/计数器为定时器 工作模式为16位定时器0模式1TH0 = 0xfc;//设定定时器0高8位初值 TL0 = 0x18;//设定定时器0低8位初值TF0 = 0;//定时器0溢出中断标志位清0ET0 = 1;//打开定时器中断开关EA = 1;//打开定时器中断总开关TR0 = 1//打开定时器0开关} void Timer0() interrupt 1//定时器0的16位定时模式1用12分频定时1ms中断函数 晶振为12MHz
{TR0 = 0;//关定时器0开关/****Timer0TimeCount++;//定时器0定时计数自加if(Timer0TimeCount >= 10)//10ms时间到{Timer0TimeCount = 0;//定时器0定时计数清0DAC0832Change(NumberValue);//DAC0832转化函数 }****///KeyScan();//按键扫描函数NixieTubeDisplayData();//数码管显示数据函数  TH0 = 0xfc;//设定定时器0高8位初值TL0 = 0x18;//设定定时器0低8位初值TR0 = 1;//开定时器0开关} void Timer1Init()//定时器1的8位自动重装模式2用12分频定时1us初始化函数 晶振为12MHz
{AUXR &= 0xbf;//设定定时器/计数器模式为12TTMOD &= 0x0f;//设定定时器/计数器工作模式清0TMOD |= 0x20;//设定定时器/计数器为定时器 工作模式为8位自动重装模式2TH1 = 0xff;//设定定时器0高8位初值 TL1 = 0xff;//设定定时器0低8位初值TF1 = 0;//定时器0溢出中断标志位清0ET1 = 1;//打开定时器中断开关EA = 1;//打开定时器中断总开关TR1 = 1//打开定时器0开关} void Timer1() interrupt 3//定时器1的8位自动重装模式2用12分频定时1us中断函数 晶振为12MHz
{ADC0809CLK = !ADC0809CLK;//500KHz的时钟} void main()//主函数
{uchar AnalogDataResult;//声明模拟数字结果变量uchar AnalogSamplingCount;//声明模拟采样计数变量uint AnalogFilterVoltage;//声明模拟滤波电压变量Timer0Init();//定时器0的16位定时模式1用12分频定时1ms初始化函数 晶振为12MHzTimer1Init();//定时器1的8位自动重装模式2用12分频定时1us初始化函数 晶振为12MHz//NumberValueSet();//数字量数值设置函数while(1)//主循环{AnalogDataResult = ADC0809Change();//ADC0809转化函数转化的模拟数据赋给模拟数据变量OutPutVoltage = (AnalogDataResult*1.0*5/255)*1000;//输出电压计算公式 5是基准电压5V 255是模数芯片ADC0809内部八位模拟转换寄存器储存的最大数值 为啥乘以1000?由于输出电压是用四位数码管来显示 需要乘以1000来把输出电压变成四位数在四位数码管上分解显示出来AnalogFilterVoltage = AnalogFilterVoltage + OutPutVoltage;//模拟滤波电压变量AnalogSamplingCount++;//模拟采样计数变量自加1if(AnalogSamplingCount >= 8)//模拟采样计数变量计8次{AnalogFilterOutPutVoltage = AnalogFilterVoltage >> 3;//模拟滤波电压变量右移三位 表示模拟滤波电压变量除以8取平均滤波后的输出电压AnalogSamplingCount = 0;//模拟采样计数变量清0AnalogFilterVoltage = 0;//模拟滤波电压变量清0  }NixieTubeDisplayDataSplit()//数码管显示数据分解函数}} 

这篇关于基于STC12C5A60S2系列1T 8051单片的模数芯片ADC0809实现模数转换应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/421214

相关文章

SpringShell命令行之交互式Shell应用开发方式

《SpringShell命令行之交互式Shell应用开发方式》本文将深入探讨SpringShell的核心特性、实现方式及应用场景,帮助开发者掌握这一强大工具,具有很好的参考价值,希望对大家有所帮助,如... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定

SpringBoot应用中出现的Full GC问题的场景与解决

《SpringBoot应用中出现的FullGC问题的场景与解决》这篇文章主要为大家详细介绍了SpringBoot应用中出现的FullGC问题的场景与解决方法,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录Full GC的原理与触发条件原理触发条件对Spring Boot应用的影响示例代码优化建议结论F

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

MySQL 分区与分库分表策略应用小结

《MySQL分区与分库分表策略应用小结》在大数据量、复杂查询和高并发的应用场景下,单一数据库往往难以满足性能和扩展性的要求,本文将详细介绍这两种策略的基本概念、实现方法及优缺点,并通过实际案例展示如... 目录mysql 分区与分库分表策略1. 数据库水平拆分的背景2. MySQL 分区策略2.1 分区概念

OpenCV图像形态学的实现

《OpenCV图像形态学的实现》本文主要介绍了OpenCV图像形态学的实现,包括腐蚀、膨胀、开运算、闭运算、梯度运算、顶帽运算和黑帽运算,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起... 目录一、图像形态学简介二、腐蚀(Erosion)1. 原理2. OpenCV 实现三、膨胀China编程(

通过Spring层面进行事务回滚的实现

《通过Spring层面进行事务回滚的实现》本文主要介绍了通过Spring层面进行事务回滚的实现,包括声明式事务和编程式事务,具有一定的参考价值,感兴趣的可以了解一下... 目录声明式事务回滚:1. 基础注解配置2. 指定回滚异常类型3. ​不回滚特殊场景编程式事务回滚:1. ​使用 TransactionT

Android实现打开本地pdf文件的两种方式

《Android实现打开本地pdf文件的两种方式》在现代应用中,PDF格式因其跨平台、稳定性好、展示内容一致等特点,在Android平台上,如何高效地打开本地PDF文件,不仅关系到用户体验,也直接影响... 目录一、项目概述二、相关知识2.1 PDF文件基本概述2.2 android 文件访问与存储权限2.

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求

Spring Shell 命令行实现交互式Shell应用开发

《SpringShell命令行实现交互式Shell应用开发》本文主要介绍了SpringShell命令行实现交互式Shell应用开发,能够帮助开发者快速构建功能丰富的命令行应用程序,具有一定的参考价... 目录引言一、Spring Shell概述二、创建命令类三、命令参数处理四、命令分组与帮助系统五、自定义S

SpringBatch数据写入实现

《SpringBatch数据写入实现》SpringBatch通过ItemWriter接口及其丰富的实现,提供了强大的数据写入能力,本文主要介绍了SpringBatch数据写入实现,具有一定的参考价值,... 目录python引言一、ItemWriter核心概念二、数据库写入实现三、文件写入实现四、多目标写入