AD转化
咱们先看看R1和R2,R2是个可调电阻假如咱们将R2变大RA1这个管脚上的电压就越大。R2变小RA1这个管脚上的电压就越小。那单片机是怎样知道电压改变的。这就需求AD转化。便是将模仿量转化成数字量。
PIC单片机怎样表明电压
PIC用十位二进制位的数来表明电压,也便是数值0~1023来表明电压。那比方现在这个数值是400那这代表多少的电压?这就要依据参阅电压来确认了。
比方咱们设置正参阅电压为3.3V ,当输入的电压为0时,数值就为0。当输入的电压为3.3V时,数值便是1023. 那假如输入的电压是1.2V代表多少电压。
首要,先算出一个数值代表多少的电压 3.3V除以1023 约等于 0.003V 。
然后,1.2V除以0.003V 等于400. 这就得出了400代表的是1.2V。
见下图咱们能够看AN0~AN7.这些都是能够装备成模仿输入的端口。只要这些引脚才干做为AD转化的端口。
实例解说:
例如: 咱们看第一张的原理图,从RA0/AN0脚输入个模仿量假如电压大于1.2v则LED亮不然LED灭。
AD的设置过程:
1,设置端口
将RA0口设置为输入TRISA = 0x01;
将RA0口设置为模仿ANSELA = 0x01;
2, 装备ADC模块
挑选ADC的转化时钟。
怎样挑选转化时钟呢 要依据现在的时钟频率进行挑选。能够依据数据手册中的表格进行挑选 。
咱们设置单片机的时钟频率为32MHZ ,挑选ADC周期要害不要挑选暗影部分,在32MHz 这一列 咱们随意挑选了ADC时钟周期1us,对应的时钟源为Fosc/32.,AD操控寄存器1 ADCON1的ADCS《2:0》=010注:ADCS《2:0》代表的意思便是 ADCS的0到2位
装备参阅电压
咱们这儿把正参阅电压装备为电源压。AD操控寄存器1ADCON1的ADPREF《1:0》=00;
装备左/右对齐
AD转化后数值是十位的二进制,咱们用单片机却仅仅八位的,所以PIC单片机,用两个八位的寄存器来寄存AD值,ADRESH用来寄存高位成果,ADRESL用来寄存低位成果。但是ADRESH和ADRESL加起来是十六啊。那这十位的数值是怎样放在里边的。这就靠左右对齐来设置,
假如是右对齐 低8八位放在ADRESL,剩余的2位放在ADRESH中。
假如是左对齐 高8八位放在ADRESH,剩余的2位放在ADRESL中。见下图
咱们这儿挑选右对齐,所以AD操控寄存器1ADCON1的ADFM=1
上面将有关ADCON1寄存器的装备说完了。下面来解说ADCON0
挑选ADC输入通道
AD转化模块只要一个,而AD输入通道有8个AN0~AN7.所以不可能一起进行AD转化,那个需求用咱们就分配给那个,依据硬件咱们将AD转化模块分配给AN0.
所以 ADCON0 的CHS《4:0》=0000;
敞开ADC模块
ADC模块敞开,ADCON0的ADON=1,仅仅单纯的启用ADC模块。并不开端AD转化。假如不必ADC模块时分主张封闭。能够省点电哦!!!
3 开端AD转化
ADCON0的GO/DONE=1敞开AD转化。
4 等候AD转化完毕
5 读取成果
一般情况下咱们并不取一次的AD转化的值。而是取屡次之后算平均值。这样来保证转化的准确性。装备ADC模块,有许多地方并没有解说为什么这么装备,由于许多装备其实是比较随意的。并不是那么的肯定的。必定非要挑选哪一个。当然实践的装备仍是要依据你项目需求。
//开发环境MPLAB X IDE ,单片机PIC16LF1823.
#include
__CONFIG(FOSC_INTOSC&WDTE_OFF&PWRTE_ON&MCLRE_OFF&CP_ON&CPD_OFF&BOREN_ON
&CLKOUTEN_OFF&IESO_ON&FCMEN_ON);//这个要放到上一行去
__CONFIG(PLLEN_OFF&LVP_OFF) ;
#define ADC_NUM 8 //转化的次数
#define LED LATA1
void init_GPIO(void)
{
TRISA = 0x01;//端口设置为输入
ANSELA = 0x01;//设置为模仿输入
PORTA = 0x00;
LATA = 0x00;
}
void init_fosc(void)
{
OSCCON = 0xF0;//32MHZ
}
void init_AD(void)
{
ADCON1= 0xA0;//右对齐,AD时钟为Fosc/32,参阅电压为电源电压,
ADCON0= 0x00;//挑选通道AN0
ADCON0bits.ADON = 1;//敞开模块
}
unsigned int ADC_BAT_ONE(void)//转化一次
{
unsigned int value;
value=0;
ADCON0bits.CHS =0;//挑选通道AN0
ADCON0bits.ADGO=1;//开端转化
while(ADCON0bits.GO==1);//等候转化完毕
value=(unsigned int)ADRESH;//强制类型转化,由于ADRESH是字符型的只能表明8位二进制。所以有必要转化成能够包容10位二进制的整型。
value= value《《8;// 将高两位左移8位
value += ADRESL;//低八位参加ADRESL的值。
return value;
}
unsigned int ADC_BAT_conTIue(void)
{
unsigned int ADV_MCU[ADC_NUM],ADV_CNT,ADV_ALL;
ADV_ALL=0;
for(ADV_CNT=0;ADV_CNT {
ADV_MCU[ADV_CNT]=ADC_BAT_ONE();
}
for(ADV_CNT=0;ADV_CNT {
ADV_ALL += ADV_MCU[ADV_CNT];
}
ADV_ALL= ADV_ALL/ADC_NUM;
return ADV_ALL;//得到成果回来
}
/*
*
*/
int main(int argc, char** argv) {
init_fosc();//设置时钟
init_AD();//设置AD
while(1)
{
if( ADC_BAT_conTIue()》400)//判别输入电压是否大于1.2V
{
LED=1;//灯亮
}
else
{
LED=0;//灯灭
}
}
}
责任编辑;zl