单片机和ADC0832的AD模数转换

发布者:橙子1234最新更新时间:2015-09-23 来源: eefocus关键字:单片机  ADC0832  模数转换 手机看文章 扫描二维码
随时随地手机看文章
在工业控制和智能化仪表中,通常由微型计算机进行实时控制及实时数据处理。计算机所加工的信息总是数字量,而被控制或被测量的有关参量往往是连续变化的模拟量,如温度、速度、压力等等,与此对应的电信号是模拟信号。模拟量的存储和处理比较困难,不适合作为远距离传输且易受干扰。在一般的工业应用系统中传感器把非电量的模拟信号变成与之对应的模拟信号,然后经模拟(Analog)到数字(Digital)转换电路将模拟信号转成对应的数字信号送微机处理。这就是一个完整的信号链,模拟到数字的转换过程就是我们经常接触到的ADC(Analog to Digital Convert)电路。

模-数转换(ADC)简介

模-数转换原理

    ADC的转换原理根据ADC的电路形式有所不同。  ADC电路通常由两部分组成,它们是:采样、保持电路和量化、编码电路。其中量化、编码电路是最核心的部件,任何ADC转换电路都必须包含这种电路。 ADC电路的形式很多,通常可以并为两类:

    间接法:它是将采样-保持的模拟信号先转换成与模拟量成正比的时间或频率,然后再把它转换为数字量。这种通常是采用时钟脉冲计数器,它又被称为计数器式。它的工作特点是:工作速度低,转换精度高,抗干扰能力强。

    直接法:通过基准电压与采样-保持信号进行比较,从而转换为数字量。它的工作特点是:工作速度高,转换精度容易保证。 

    模—数转换的过程有四个阶段,即采样、保持、量化和编码。

    采样是将连续时间信号变成离散时间信号的过程。经过采样,时间连续、数值连续的模拟信号就变成了时间离散、数值连续的信号,称为采样信号。采样电路相当于一个模拟开关,模拟开关周期性地工作。理论上,每个周期内,模拟开关的闭合时间趋近于0。在模拟开关闭合的时刻(采样时刻),我们就“采”到模拟信号的一个“样本”。

    量化是将连续数值信号变成离散数值信号的过程。理论上,经过量化,我们就可以将时间离散、数值连续的采样信号变成时间离散、数值离散的数字信号。

    我们知道,在电路中,数字量通常用二进制代码表示。因此,量化电路的后面有一个编码电路,将数字信号的数值转换成二进制代码。

    然而,量化和编码总是需要一定时间才能完成,所以,量化电路的前面还要有一个保持电路。保持是将时间离散、数值连续的信号变成时间连续、数值离散信号的过程。在量化和编码期间,保持电路相当于一个恒压源,它将采样时刻的信号电压“保持”在量化器的输入端。虽然逻辑上保持器是一个独立的单元,但是,工程上保持器总是与采样器做在一起。两者合称采样保持器。

八位串行A/D转换器ADC0832简介

    ADC0832 是美国国家半导体公司生产的一种8 位分辨率、双通道A/D转换芯片。由于它体积小,兼容性强,性价比高而深受单片机爱好者及企业欢迎,其目前已经有很高的普及率。ADC083X是市面上常见的串行模—数转换器件系列。ADC0831、ADC0832、ADC0834、ADC0838是具有多路转换开关的8位串行I/O模—数转换器,转换速度较高(转换时间32uS),单电源供电,功耗低(15mW),适用于各种便携式智能仪表。本章以ADC0832为例,介绍其使用方法。

    ADC0832是8脚双列直插式双通道A/D转换器,能分别对两路模拟信号实现模—数转换,可以用在单端输入方式和差分方式下工作。ADC0832采用串行通信方式,通过DI 数据输入端进行通道选择、数据采集及数据传送。8位的分辨率(最高分辨可达256级),可以适应一般的模拟量转换要求。其内部电源输入与参考电压的复用,使得芯片的模拟电压输入在0~5V之间。具有双数据输出可作为数据校验,以减少数据误差,转换速度快且稳定性能强。独立的芯片使能输入,使多器件挂接和处理器控制变的更加方便。

    ADC0832 具有以下特点:

· 8位分辨率;

· 双通道A/D转换;

· 输入输出电平与TTL/CMOS相兼容;

· 5V电源供电时输入电压在0~5V之间;

· 工作频率为250KHZ,转换时间为32μS;

· 一般功耗仅为15mW;

· 8P、14P—DIP(双列直插)、PICC 多种封装;

· 商用级芯片温宽为0°C to +70°C,工业级芯片温宽为-40°C to +85°C;

adc0832

图3 ADC0832引脚图

芯片接口说明:

· CS_ 片选使能,低电平芯片使能。

· CH0 模拟输入通道0,或作为IN+/-使用。

· CH1 模拟输入通道1,或作为IN+/-使用。

· GND 芯片参考零电位(地)。

· DI 数据信号输入,选择通道控制。

· DO 数据信号输出,转换数据输出。

· CLK 芯片时钟输入。

· Vcc/REF 电源输入及参考电压输入(复用)

ADC0832的工作原理:

    正常情况下ADC0832 与单片机的接口应为4条数据线,分别是CS、CLK、DO、DI。但由于DO端与DI端在通信时并未同时使用并与单片机的接口是双向的,所以在I/O口资源紧张时可以将DO和DI并联在一根数据线上使用。当ADC0832未工作时其CS输入端应为高电平,此时芯片禁用,CLK 和DO/DI 的电平可任意。当要进行A/D转换时,须先将CS使能端置于低电平并且保持低电平直到转换完全结束。此时芯片开始转换工作,同时由处理器向芯片时钟(CLK)输入端输入时钟脉冲,DO/DI端则使用DI端输入通道功能选择的数据信号。在第一个时钟脉冲的下沉之前DI端必须是高电平,表示启始信号。在第二、三个脉冲下沉之前DI端应输入两位数据用于选择通道功能。

通道地址

通道

工作方式说明

SGL/DIF

ODD/SIGN

0

1

0

0

+

-

差分方式

0

1

-

+

1

0

+

 

单端输入方式

1

1

 

+

表1:通道地址设置表

    如表1所示,当此两位数据为“1”、“0”时,只对CH0 进行单通道转换。当2位数据为“1”、“1”时,只对CH1进行单通道转换。当两位数据为“0”、“0”时,将CH0作为正输入端IN+,CH1作为负输入端IN-进行输入。当两位数据为“0”、“1”时,将CH0作为负输入端IN-,CH1 作为正输入端IN+进行输入。到第三个脉冲的下降之后DI端的输入电平就失去输入作用,此后DO/DI端则开始利用数据输出DO进行转换数据的读取。从第4个脉冲下降沿开始由DO端输出转换数据最高位Data7,随后每一个脉冲的下降沿DO端输出下一位数据。直到第11个脉冲时发出最低位数据Data0,一个字节的数据输出完成。也正是从此位开始输出下一个相反字节的数据,即从第11个字节的下降沿输出Data0。随后输出8位数据,到第19 个脉冲时数据输出完成,也标志着一次A/D转换的结束。最后将CS置高电平禁用芯片,直接将转换后的数据进行处理就可以了。时序说明请参照图4。

    作为单通道模拟信号输入时ADC0832的输入电压是0—5V且8位分辨率时的电压精度为19.53mV,即(5/256)V。如果作为由IN+与IN-输入的输入时,可是将电压值设定在某一个较大范围之内,从而提高转换的宽度。但值得注意的是,在进行IN+与IN-的输入时,如果IN-的电压大于IN+的电压则转换后的数据结果始终为00H。

ADC0832的工作时序

adc0832

图4 ADC0832工作时序

ADC0832软硬件设计实例

    通过以上的理论学习之后,对模—数转换应该有了一定的了解,接下来就根据上文的指导,对ADC0832进行实际应用,以加深印象。本实例功能是将通道1上采样到的电压显示在LED数码管上,通过改变通道1的输入电压变化,观察输出读数。

硬件原理图

sch

图7 硬件原理图

程序流程图

software

图8 软件流程图

    相信看到这里,你应该可以理解我们是如何利用单片机来进行模数转换的处理了,你也可以根据自己的需要来写些AD模数转换相关的应用程序,如数字温度计,湿度传感应用,压力传感应用等等。由于篇幅有限,读者朋友可以通过网站或电子邮件一起交流与学习。在下几期中,我们将陆续介绍51单片机综合学习系统的其它功能原理与应用。[page]

STC单片机的相关程序如下:

#include     //定义的 系统头文件和全局变量
#include

#define uchar unsigned char
#define uint unsigned int
#define DogReset()    WDT_CONTR=0x35

// T1 定时 0.1ms.作为系统计时用,
#define vT01ms    2
#define vT10ms    10
#define vT100ms    10
#define vT01S    100  // 1 s = 10 ms * 100
#define vT0HVal    0xfe //0xff //0xfe //0xf6
#define vT0LVal    0x33 //0x9c //0x0c //0x4c


uchar code display_AD_channel_ID[2] = {0x00,0x01};

static unsigned char data CS;

uchar data AD_channel_result[2][5]; //各通道A/D转换结果。前是通道号;后是转换的值

uint cT01ms;
uchar cT10ms;
uchar cT100ms;
uchar cT01s;
uchar  THTL;

bit  OutFlag;

void delay_ms(register uint Count){
 register uchar T;

 for(;Count>0;Count--){
  for(T=0;T<80;T++){
   _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
   _nop_(); _nop_(); _nop_(); _nop_(); _nop_();
  }
  DogReset();
 }
}


void send_char_com(unsigned char OutData){
 SBUF = OutData; //输出字符
 while(!TI); //空语句判断字符是否发完
 TI = 0; //清TI
}

void send_string_com(uchar *str,uchar strlen){
 uchar i;
 for(i=strlen; i>0; i--){
  send_char_com(*str);
  str++;
  DogReset();
 }
}


uchar Ad_Change(uchar channel){
 uint AD_Result_Temp = 0 ;
 //---------------------将P1.0--P1.1设置成适合AD转换的模式
 /// P1 = 0xff;                 //将P1口置高,为A/D转换作准备
 ADC_CONTR = ADC_CONTR|0x80; //1000,0000打开A/D转换电源
 P1M0 = 0x03;                //0000,0011用于A/D转换的P1.x口,先设为开漏
 P1M1 = 0x03;                //0000,0011P1.0--P1.1先设为开漏。断开内部上拉电阻
 delay_ms(20);                   //20

 ADC_CONTR = ADC_CONTR&0xE0; //1110,0000 清ADC_FLAG,ADC_START位和低3位
 ADC_CONTR = ADC_CONTR|(display_AD_channel_ID[channel]&0x07); //设置当前通道号

 delay_ms(1);                 //延时使输入电压达到稳定
 ADC_DATA = 0;               //清A/D转换结果寄存器
 ADC_LOW2 = 0;

     ADC_CONTR = ADC_CONTR|0x08; //0000,1000ADCS = 1,启动转换

 do { DogReset();}
     while((ADC_CONTR & 0x10)==0);   //0001,0000等待A/D转换结束


 ADC_CONTR = ADC_CONTR&0xE7; //1110,0111清ADC_FLAG位,停止A/D转换

 AD_Result_Temp = ((AD_Result_Temp|ADC_DATA)<<2)|(ADC_LOW2&0x03);
 //保存返回AD转换的 结果
 //----------------------------转换成可由串口显示的字符
 AD_channel_result[channel][0] = AD_Result_Temp/1000+0x30;
 AD_channel_result[channel][1] = (AD_Result_Temp%1000)/100+0x30;
 AD_channel_result[channel][2] = (AD_Result_Temp%100)/10+0x30;
 AD_channel_result[channel][3] = AD_Result_Temp%10+0x30;


 //------------------------串口监视
//  send_char_com(ADC_DATA);    //////发送转换 的 到的 值,这里只是 高8位,值的转换需要考虑
 // send_char_com(ADC_LOW2);    //////发送转换 的 到的 值,这里只是 低2位,值的转换需要考虑

// send_string_com(AD_channel_result[channel],4);

 delay_ms(10);    //
 return(ADC_DATA);

}

uchar AD_Filter(void){
 uchar i;
 uchar cTemp[32];
 uchar cAverage;

 for(i=32;i>0;i--){
  cTemp[i]=Ad_Change(0);
  cAverage=((cAverage+cTemp[i])>>1);
 }
 return(cAverage);
}

void InitCom(unsigned char BaudRate){
 switch (BaudRate){
   case 1:  THTL = 64; break; //波特率300
   case 2:  THTL = 160; break; //600
   case 3:  THTL = 208; break; //1200
   case 4:  THTL = 232; break; //2400
   case 5:  THTL = 244; break; //4800
   case 6:  THTL = 250; break; //9600
   case 7:  THTL = 253; break; //19200
   case 8:  THTL = 255; break; //57600
   default:  THTL = 208; break; //1200
 }
}


void Chip_initial(void){

 IE=0;
 // 定时器控制字初始化
 TMOD=0x21;      // 定时器1为方式2,定时器0为方式1
 TCON=0x50;      // 设置外部中断类型
 T2CON=0x0d;      // 选择定时器1为波特率发生器,T2为捕获工作方式
// ET2=1;
 ET0=1;      // 允许定时器0、定时器2中断


 // 外部中断设置
 EX0=0;       // FFSK中断初始时关闭,有载波时再开启
 EX1=0;
 IT1=1;       // 外部中断0、1均为下降沿触发
 IT0=1;

 // 启动定时器0
 TH0=vT0HVal;      // 启动定时器0
 TL0=vT0LVal;
 TR0=1;

 InitCom(6); //设置波特率为9600 1-7波特率300-19200
  SCON = 0x50; //串口方式1,允许接收
 TH1 = THTL;
 TL1 = THTL;
 PCON = 0x80;  //波特率加倍控制,SMOD位
 RI = 0;   //清收发标志
 TI = 0;
 TR1 = 1;  //启动定时器

 IP=0x02;     //PT2=1
 IPH=0x02;     //PT2H=1,PT0H=1
 EA=1;

 delay_ms(10);     // 延时是为了避免定时器0无法产生中断的问题
}


void Para_initial(void){

 OutFlag=0;

 cT01ms  vT01ms;
 cT10ms  vT10ms;
// cT100ms  =  vT100ms;
 cT01s  =  vT01S;

 CS = 0;   //设置CS为0不选任何的音源,如为1则选第1路
}


void system_initial(void){
 Chip_initial();
 Para_initial();
 DogReset();   // 已针对 STC89C58RD+ 作修改.06-04-06
}


void main(void){

 system_initial();

 while(1){
  DogReset();
  if(OutFlag){
   OutFlag=0;
   send_char_com(AD_Filter());
  }
 }
}


void Trint0(void) interrupt 1 using 1{

 TR0=0;          // 时基1mS
 TH0=vT0HVal;
 TL0=vT0LVal;
 TR0=1;
 TF0=0;

 if(!(--cT01ms)){
  cT01ms=vT01ms;
  if(!(--cT10ms)){
    cT10ms=vT10ms;
   if(!(--cT01s)){
    cT01s=vT01S;
    OutFlag=1;

   }
  }
 }
}

 

void ComInINT(void) interrupt 4 {
 if (RI){ //判断是不收完字符
  switch(SBUF){
   case 0x61: CS = 1; break; //根据SBUF设置CS 接收'abcde'调试方便
   case 0x62: CS = 2; break;
   case 0x63: CS = 3; break;
   case 0x64: CS = 4; break;
   case 0x65: CS = 0; break;
  }
  P1 = 255; //P1口全为高电平,4-7通过反相为低不选任何音源,0-3为高用于读取按键
  RI = 0; //RI清零
 }
}

关键字:单片机  ADC0832  模数转换 引用地址:单片机和ADC0832的AD模数转换

上一篇:单片机的罢工
下一篇:MC145152+MC1648+MC12022锁相环的制作

推荐阅读最新更新时间:2024-03-16 14:32

51单片机I2C详解与程序源码
I2C是由Philips公司发明的一种串行数据通信协议,仅使用两根信号线:SerialClock(简称SCL)和SerialData(简称SDA)。I2C是总线结构,1个Master,1个或多个Slave,各Slave设备以7位地址区分,地址后面再跟1位读写位,表示读(=1)或者写(=0),所以我们有时也可看到8位形式的设备地址,此时每个设备有读、写两个地址,高7位地址其实是相同的。 I2C数据格式如下: 无数据:SCL=1,SDA=1; 开始位(Start):当SCL=1时,SDA由1向0跳变; 停止位(Stop):当SCL=1时,SDA由0向1跳变; 数据位:当SCL由0向1跳变时,由发送方控制SDA,此时SDA为有效数据
[单片机]
充分利用串行口数据传输优势实现单片机通信设计
1.引言 MCS51 单片机 标准构成只有一组全双工UART串行口,P3.0-RXD收、P3.1-TXD发,如果要完成多路串口收发,一般使用外部扩展芯片,例如:GM8123/25,完成一扩多路串口,但是,由于增加扩展芯片,造成硬件成本增加,PCB板面增大,抗干扰性下降等不良因素,因此能尽量使用现有串口,利用分时切换技术,完成串行口不同类型,不同端口的数据传送是优先选择。 2.硬件功能简述 本设计用于数据采集监控系统如图1所示。采用485串行口完成数据采集,和232串口输出打印功能,这样使得接口连线简单,可靠性高。 RS-485工作特征:发送端:逻辑“1″两线间的电压差为+2至6V表示;逻辑”0″以两线间的电压差为-2至
[电源管理]
充分利用串行口数据传输优势实现<font color='red'>单片机</font>通信设计
试析电子技术中的单片机应用
一、单片机的特点与基本组成 与其它的嵌入式系统相比,单片机的体积小,但是集成度高,具备较高的可靠性与控制功能;功耗低且采用低电压,因此对便携式产品的制造与生产十分有利;具备较好的扩展性与优异的性能比,其应用范围十分广泛,包括办公室自动化设备、实时过程的控制、各类仪器仪表、医疗领域相关设备、汽车电子产品以及计算机网络通信技术等等,由此可见,单片机是一种实用性非常强的嵌入式系统。其基本组成包括以下几个部分:第一,运算器,其核心是ALU部件,主要作用就是完成二进制算术与逻辑运算,运算器的辅助设备包括暂存器TMP、寄存器B、累加器 ACC、布尔处理器以及程序状态标志寄存器 PSW 等等;第二,控制器,其为CPU的神经中枢,包括定时控制逻辑
[电源管理]
利用超级终端实现单片机与电脑通讯
利用超级终端实现单片机与电脑通讯 超级终端的设置步骤如上图所示,下面是单片机的硬件连接电路: 单片机C程序: #include reg52.h void int_initial()//串口中断初始化 { EA=1; ES=1; } void RS232_receive_Init() //串口初始化设置 { SCON = 0x50; // UART工作于模式1, REN=1 TMOD = 0x20 ; // Timer1 工业于模式2 PCON |= 0x80; //
[单片机]
利用超级终端实现<font color='red'>单片机</font>与电脑通讯
LCD1602液晶显示模块的单片机驱动深入详解之软件篇(AVR)
LCD1602液晶显示模块的驱动虽然比七段数码管之类的显示要复杂一些,但实际上也并不是很难,最主要的还是初始化,为什么这么说呢?我们在调试一块新液晶屏的时候,都会先初始化看看有没有光标在闪,没有光标前是一番努力(PROTEUS上也是这么做的),光标出来之后就相对很容易了,因为光标出来了,至少说明硬件连接是没有问题的,模块也已经成功接收到了指令,后面就是啃数据手册、改程序、烧录程序观察显示的循环了。 那初始化的流程是怎么样的呢?我们还是看看HD44780的数据手册吧,如下图所示: 可以看到,初始化的主要步骤如下: (1)上电:这特么也算是一个步骤么?是的!如果你用的是其它液晶模块,比如LCD12864,会发现有一个复位引脚,LC
[单片机]
LCD1602液晶显示模块的<font color='red'>单片机</font>驱动深入详解之软件篇(AVR)
一种基于单片机和串行EEPROM的智能密码锁
1. 引言 电子技术的飞速发展,给传统的机械锁带来了巨大的变革,现代的电子技术与机械技术相结合,产生了一大批如声控锁、电子密码锁、遥控锁,指纹锁等先进的锁具。虽然这类产品安全性高,但因其生产成本高,安装使用不方便,在一定程度上限制了这类产品的普及和推广。本文介绍的是一种基于AT89C51单片机和AT24C01串行EEPROM的智能密码锁的硬件设计和软件实现方法,这种电路设计具有防试探按键输入、智能控制上锁、开锁、报警、修改密码等多种功能。密码长度可变,保密性强,灵活性高,外接各种执行机构,可广泛用于车辆、大门、保险柜等各种需上锁的场合。 2. 设计思路 本设计中,智能密码锁工作时分为两种工作状态,分别是正常状态和锁定状态。锁
[单片机]
一种基于<font color='red'>单片机</font>和串行EEPROM的智能密码锁
51单片机的一些开发实用技巧
一.C语言中嵌入汇编语言 单片机开发中,通常我们使用C语言编写主程序,这样可以充分借助C语言工具提供的运算库函数及强大的数据处理能力。但C语言的可控性不及汇编语言,在有些对时序要求严格的处理上,我们还需用灵活性更强的汇编语言来编写。上海AVR单片机培训这样就产生了C语言和汇编语言混合编程的问题,一般分成三种方式:1.汇编语言调用C语言函数;2. C语言调用汇编语言;3. C语言中嵌入汇编语言。这里我们主要介绍第3种,即C语言中嵌入汇编语言。 下面的一段程序是主程序调用精确的205μS延时子程序并使P1.0交替输出高、低电平的方波。 /*------------程序名test.c------------*/ #include P
[单片机]
51<font color='red'>单片机</font>的一些开发实用技巧
stm32系列MCU芯片选型表
本文为stm32系列32位微控制器选型表,包括STM32F0系列、STM32L1系列、STM32F1系列、STM32F2系列、STM32F3系列、STM32F4系列选型,STM32系列36脚、48脚、64脚、100脚、144脚选型。 TM32系列36脚、48脚、64脚、100脚、144脚选型: STM32系列按内核架构分类产品选型: STM32F0 系列 STM32L1 系列 STM32F1 系列 STM32F2 系列 STM32F3 系列 STM32F4 系列
[单片机]
stm32系列<font color='red'>MCU</font>芯片选型表
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved