十四、51单片机之AD转换

1、AD相关简介

1.1、什么是AD转换?

(1)A是指analog、模拟的;D是指digital、数字的。

(2)现实世界是模拟的,连续分布的,无法被分成有限份;计算机世界是数字的,离散分布的,是可以被分成有限份的;

(3)AD转换就是把一个物理量从模拟的转换成数字的。

1.2、AD转换的意义

自然界多数是模拟量,为了方便计算机处理,人为的数字化了自然界的模拟量。

1.3、STC89C516

该单片机本身引脚没有模拟的,所以模拟量不能直接给该单片机。后续很多新的单片机都具有模拟输入输出的引脚,引脚会对应相应的AD转换。

1.4、AD转换原理

(1)采用比较器电路进行AD转换。

1.5、AD转换中的主要概念

(1)位数: AD转换后转出来的数由几位二进制来表示。位数越多,越细腻,精度越高。

(2)量程:AD转换器可以接受的模拟量的范围。

(3)精度:简单理解就是转出来到底有多准

(4)分辨率:AD转换器转出来的二进制数,每一格表示多少

(5)转换速率(转换时间)

1.6、AD转换在系统中存在的方式

(1)CPU外部扩展专用AD芯片

(2)CPU内部集成AD模块(内部外设)

2、原理图

2.1、简介

(1)ET2046是我们外接的AD转换芯片。

(2)AIN0、AIN1、AIN2为三路AD转换。

(3)AD1502是一个滑动变阻器;NTC1是一个热敏电阻;GR1是一个光敏电阻。

2.2、接线

(1)CLK 接P1.0、T-CS接P1.1、DI接P1.2、DOUT接P1.3。

(2)CLK为SPI时钟线。

(3)T-CS为使能线。

(4)DI是数字输入线,DOUT是数字输出线。

3、ET2046数据手册

该芯片已停产,官网没搜到,数据手册链接:ET2046 - 百度文库

(1)ET2046是4线触摸屏控制器,在此用于AD转换。ET2046是一个经典的逐次逼近寄存器模数转换器。

(2)转换器模拟通道:X-、Y-、Z坐标、辅助输入(AUX)、电池电压(VBAT)、芯片温度。通过一个多路选择器提供。

(3)控制字

bit7固定为1
bit6-4A2-A0,选择模拟通道
bit3设置采样位数。0表示12bit,1表示8bit
bit2为1表示用单端模式,为0表示差分模式。此处用单端模式。
bit1-0power down模式使能,00表示使能。

4、示例程序

通过ET2046将模拟电压值转换位12bit数字量,并通过串口以文本方式显示。

 (1)app.c文件

#include <reg51.h>
#include "et2046.h"
#include "drv_uart.h"

void Delay500ms();		//@12MHz

void main()
{
	uint AD_Value = 0;   /*AD值*/
	uchar AD_H = 0;      /*AD值高4位*/
	uchar AD_L = 0;      /*AD低8位*/
	UartInit();           /*串口初始化函数*/
	while(1)
	{
		AD_Value = Read_AD_Data(0x94);    /*滑动变阻器*/
		
		/*16进制显示采集的12bitAD值*/
		//AD_H = AD_Value >> 8;
		//AD_L = AD_Value & 0xff;  /*按位与*/
		//UartSendByte(AD_H);
		//UartSendByte(AD_L);
		
		/*文本形式显示采集的AD值,0xFFF = 4095*/
		UartSendString("AD_Value:");
		UartSendByte((AD_Value / 1000) + 48);       /*显示千位*/
		UartSendByte((AD_Value % 1000 /100) + 48);  /*显示百位*/
		UartSendByte((AD_Value % 100 /10) + 48);    /*显示十位*/
		UartSendByte((AD_Value % 10 ) + 48);        /*显示个位*/
		UartSendString(".\r\n");
		
		
		Delay500ms();
	}
}


void Delay500ms()		//@12MHz
{
	unsigned char i, j, k;

	_nop_();
	i = 4;
	j = 205;
	k = 187;
	do
	{
		do
		{
			while (--k);
		} while (--j);
	} while (--i);
}


(2)ET2046.c文件

#include "ET2046.h"


/*
 *功能:SPI写字节函数
 *参数:	Data:写入的数据
 *返回值:无
 */

void SPI_WriteByte(uchar Data)
{
	uchar i = 0;   /*用于for循环*/
	CLK = 0;
	
	
	for(i=0; i<8; i++)
	{
		DIN = Data >> 7;  	//从高位到低位写入
		Data <<= 1;
		CLK = 0;			      //上升沿放置数据
		CLK = 1;
	}
}


/*功能:SPI读12bit
 *参数:无
 *返回值:读取到的数据
 */
uint SPI_Read(void)
{
	uint i = 0;         /*用于for循环*/ 
	uint Data = 0;      /*暂存读取到的书*/
	CLK = 0;
	for(i=0; i<12; i++)	  /*接收12位数据*/
	{
		Data <<= 1;
		CLK = 1;
		CLK = 0;
		Data |= DOUT;
	}
	return Data;	
}

/*
 *功能:读取AD值  
 *参数: cmd:读取对应通道发送的控制字
 *返回值:读取的AD值
 */
uint Read_AD_Data(uchar cmd)
{
	uchar i = 0;    /*用于for循环*/
	uint AD_Value;  /*记录AD的值*/
	
	CLK = 0;
	CS  = 0;
	SPI_WriteByte(cmd);
	for(i=6; i>0; i--); 	/*延时等待转换结果*/
	
	CLK = 1;	            /*发送一个时钟周期,清除BUSY*/
	_nop_();
	_nop_();
	CLK = 0;
	_nop_();
	_nop_();
	AD_Value=SPI_Read();
	CS = 1;
	return AD_Value;	
}


(3)ET2046.h文件

#ifndef __ET2046_H__
#define __ET2046_H__

/*头文件包含*/
#include <reg51.h>
#include <intrins.h>


/*宏定义*/
#define uchar unsigned char
#define uint  unsigned int


/*定义使用的IO口*/
sbit CLK  = P1^0;	  //时钟
sbit CS   = P1^1;	  //片选
sbit DIN  = P1^2;	  //输入
sbit DOUT = P1^3;	  //输出


/*函数声明*/
void SPI_WriteByte(uchar Data);   /*SPI写字节函数*/
uint SPI_Read(void);              /*SPI读12bit*/
uint Read_AD_Data(uchar cmd);




#endif

(4)drv_uart.c文件

#include "drv_uart.h"
#include <reg51.h>


/*
 *功能:串口初始化函数,8数据位,1停止位,无校验位,波特率4800
 *参数:无
 *返回值:无
 */
void UartInit(void)
{
	SCON = 0x50;  //串口工作在模式1,8位数据位,允许串行接收
	PCON = 0x80;  //波特率加倍
	TMOD = 0x20;  //设置T1为模式2 
	TH1 = 243;    //波特率4800	,TH1 = 晶振频率/12/32/波特率
	TL1 = 243;    //8位自动重装,意识是TH1用完了之后下一个周期TL1会自动重装到TH1去。
	TR1 = 1;      //开启定时器1
	ES = 1;       //打开串口中断
	EA = 1;       //打开总中断
}


/*
 *功能:通过串口发送一个字节数据
 *参数:需要发送的内容
 *返回值:无
 */
void UartSendByte(unsigned char Dat)
{
	SBUF = Dat;      //准备好需要发送的一个字节
	while(TI == 0);  //确认串口发送没有再忙,while循环需要加超时判断
	TI = 0;					 //软件复位TI标志位
}


/*
 *功能:通过串口发送字符串
 *参数:待发送的字符串
 *返回值:无
 */
void UartSendString(unsigned char *str)
{
	while(*str != '\0')    //等待字符串发完*/
	{
		UartSendByte(*str);  //发送一个字符
		str++;               //指针指向下一个字符
	}
}


/*
 *功能:串口中断接收函数
 *参数:无
 *返回值:无
 *注意:中断函数通过中断编号识别,中断编号可通过查数据手册得到
 */
void Uart_Isr() interrupt 4 
{
	unsigned char ReceiveBit;
	if(RI == 1)
	{
		ReceiveBit = SBUF;		//读取SBUF,读取串口接收到的一个字节
		RI = 0;
	}
	//UartSendByte(ReceiveBit);		  //接收到的内容原封不动发回去
}

(5)drv_uart.h文件

#ifndef __DRV_UART_H__
#define __DRV_UART_H__


/*函数声明*/
void UartInit(void);                       /*串口初始化函数*/
void UartSendByte(unsigned char Dat);      /*通过串口发送一个字节数据*/
void UartSendString(unsigned char *str);   /*通过串口发送字符串*/

#endif

5、AD转换涉及的值

(1)AD转换芯片引脚输入一个模拟量,这个模拟值本质上讲是电压,根据AD转换的位数,电压值对应一个AD值(数字量)。

(2)AD转换输入的模拟量(电压值),有时会受一些物理量的影响。所以物理量的值对应电压值。

例如:光敏电阻光强不同,采集到的电压也会不相同。即光强度对应电压值。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

代码织梦师小猪

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值