51单片机 AD转换
在数逻的课程中,已经学习过AD转换的概念:将模拟信号采样、量化、编码后转换为数字信号。但是未学习过通过单片机编程,显示结果。
编码分有舍有入、只舍不入两种,量化误差前者更小。=2Vm/(2^n+1 - 1 )
注意,为了达到精确度高、稳定性好的目的,最好将所有器件的模拟地和数字分别连接,最后将模拟地和数字地仅在一点相连。
此处,使用的是STC12C5A60S2内部的AD转换。
/* 功能:使用12C5A60S2内部AD读取外部电压,显示在1602上 */
#include "STC12C5A60S2.H"
#include <intrins.h>
sbit RS = P2^6; //1602定义口 //HZ:EN=P2.2 RS=P2.0 RW=P2.1
sbit RW = P2^5;
sbit EN = P2^7;
#define uchar unsigned char;
#define uint unsigned int;
#define RS_CLR RS=0
#define RS_SET RS=1
#define RW_CLR RW=0
#define RW_SET RW=1
#define EN_CLR EN=0
#define EN_SET EN=1
#define DataPort P0 //连接1602数据口 P0
uchar da1=0,da2=0,da3=0;
double Data,c;
char a[5]="";
uchar ADC_Chanul_Turn=0;
void DelayUs2x(unsigned char t)
{
while(--t);
}
void DelayMs(unsigned char t)
{
while(t--)
{
//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);
}
}
/*------------------------------------------------
判忙函数
------------------------------------------------*/
bit LCD_Check_Busy(void)
{
DataPort= 0xFF;
RS_CLR;
RW_SET;
EN_CLR;
_nop_();
EN_SET;
return (bit)(DataPort & 0x80);
}
/*---------
------------------------------------------------*/
void LCD_Write_Com(unsigned char com)
{
while(LCD_Check_Busy()); //忙则等待
RS_CLR;
RW_CLR;
EN_SET;
DataPort= com; //
_nop_();
EN_CLR;
}
/*------------------------------------------------
写入数据函数
------------------------------------------------*/
void LCD_Write_Data(unsigned char Data)
{
while(LCD_Check_Busy()); //忙则等待
RS_SET;
RW_CLR;
EN_SET;
DataPort= Data;
_nop_();
EN_CLR;
}
/*------------------------------------------------
清屏函数
------------------------------------------------*/
void LCD_Clear(void)
{
LCD_Write_Com(0x01);
DelayMs(5);
}
/*------------------------------------------------
写入字符串函数
------------------------------------------------*/
void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s)//y为行数,x为横坐标,最后一个是字符
{
if (y == 0)
{
LCD_Write_Com(0x80 + x); //表示第一行
}
else
{
LCD_Write_Com(0xC0 + x); //表示第二行
}
while (*s)
{
LCD_Write_Data( *s);
s ++;
}
}
/*------------------------------------------------
写入字符函数
------------------------------------------------*/
void LCD_Write_Char(unsigned char x,unsigned char y,unsigned char Data)
{
if (y == 0)
{
LCD_Write_Com(0x80 + x);
}
else
{
LCD_Write_Com(0xC0 + x);
}
LCD_Write_Data( Data);
}
/*------------------------------------------------
初始化函数
------------------------------------------------*/
void LCD_Init(void)
{
LCD_Write_Com(0x38); /*显示模式设置*/
DelayMs(5);
LCD_Write_Com(0x38);
DelayMs(5);
LCD_Write_Com(0x38);
DelayMs(5);
LCD_Write_Com(0x38);
LCD_Write_Com(0x08); /*显示关闭*/
LCD_Write_Com(0x01); /*显示清屏*/
LCD_Write_Com(0x06); /*显示光标移动设置*/
DelayMs(5);
LCD_Write_Com(0x0C); /*显示开及光标设置*/
}
/*------------------------------------------------
ADC函数
------------------------------------------------*/
void InitADC()//初始AD寄存器
{
P1ASF=0x03; //0xff设置P1口全部为ADC通道,P10 P11为输出口
ADC_RES=0x00; //清除高8位缓冲数据
if(ADC_Chanul_Turn%2==0)
{
ADC_CONTR=0xF0; //P10口
_nop_();
_nop_();
_nop_();
_nop_();
ADC_CONTR=0xE8;
}
if(ADC_Chanul_Turn%2==1) //P11口
{
ADC_CONTR=0xF1;
_nop_();
_nop_();
_nop_();
_nop_();
ADC_CONTR=0xE9;
}
}
void timer0() interrupt 1 //interrupt 1: 定时器0,interrupt3:定时器3
{
TH0=(65536-20000)/256; //高八位,(需要表示Xms的定时,计数器由65536-X数到65536,由于16位,只能分高低位)
TL0=(65536-20000)%256; //低八位
InitADC();
}
void adc_isr() interrupt 5 //FLAG标志位置位触发中断,没有设优先级,但是同优先级下定时器0更高
{
//V_5REF=V_1REF*256/da_ref;
if(ADC_Chanul_Turn%3==0) //外部基准电压
{
da1=ADC_RES; //获取转换结果
Data=((double)da1/256)*5; //取八位计算基准电压Data,
c =Data;
}
if(ADC_Chanul_Turn%3==1)
{
da2=ADC_RES; //获取转换结果
Data=((double)da2/256)*5; //取八位计算实际值Data,
c =Data;
}
if(ADC_Chanul_Turn%3==2)
{
da3=ADC_RES; //获取转换结果
Data=((double)da3/256)*5; //取八位计算实际值Data,
c =Data;
}
a[0]=((int)c%10+0x30);//个位(电压<5,仅有个) //0x30: ASCAI码里代表“0”,必须转换成字符存在字符型数组里才可以在1602液晶屏上显示
a[1]=0x2e; //小数点
a[2]=((int)(c*10)%10+0x30); // 十分位
a[3]=((int)(c*100)%10+0x30);// 百分位
a[4]='\0'; // 加了串尾符才成了字符串哦
if(ADC_Chanul_Turn%3==0) LCD_Write_String(0,0,a);
if(ADC_Chanul_Turn%3==1) LCD_Write_String(5,0,a);
if(ADC_Chanul_Turn%3==2) LCD_Write_String(0,1,a);
ADC_CONTR&=0xEF; //标志位清零
ADC_Chanul_Turn++;
if(ADC_Chanul_Turn==252)
ADC_Chanul_Turn=0;
}
void main()
{
LCD_Init();
LCD_Clear(); //清屏
DelayMs(255);
TH0=(65536-20000)/256; //开定时器0
TL0=(65536-20000)%256;
EA=1; //开全局中断
ET0=1; //允许定时器零中断
EADC=1; //允许ADC中断
TR0=1;
while(1);
}