ADC外设的操作方式如下:
1.AD单通道,连续转换(本程序只使用AD转换器的规则组)
代码部分之模块一 void AD_Init(void)------ADC和GPIO的初始化
第一步:开启时钟
a.开启ADC的时钟和开启GPIO输入口的时钟。
b.配置ADC逐次比较过程使用的时钟。
第二步:使用结构体初始化GPIO
第三步:配置ADC规则组输入通道,以及序列号和采样频率。
a.若是AD多通道,则复制粘贴此代码,指定其它序列里的其它的通道修改参数即可。
第四步:用结构体初始化ADC(需指定扫描模式是连续扫描模式 )
第五步:ADC使能。
第六步:ADC复位校准。
第七步:选择触发源,软件触发ADC开始开始转换,因为是连续转换模式,所以只需要触发一次即可连续转换。
代码部分之模块二 uint16_t AD_GetValue(void) -----读取ADC
在函数最后返回ADC的DR寄存器的值:ADC_GetConversionValue。
代码部分之主函数:
第一步:调用ADC初始化的函数
第二步:使用while循环,调用ADC读取ADC的函数。
//主函数模块代码
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.H"
#include"AD.H"
uint16_t ADValue;
float Voltage;
int main(void)
{
AD_Init();
OLED_Init();//初始化OLED;
OLED_ShowString(1,1,"ADValue:");
OLED_ShowString(2,1,"Voltage:0.00V");
while(1)
{
ADValue=AD_GetValue();
Voltage=(float)ADValue/4095*3.3; //显示具体的AD值,将ADValue强制转换为浮点数,防止除4095变成整数
OLED_ShowNum(1,9,ADValue,4);
OLED_ShowNum(2,9,Voltage,1);
OLED_ShowNum(2,11,(uint16_t)(Voltage*100)% 100,2); //因为浮点数不能取余运算,所以取余需要用强制类型转换
Delay_ms(100);
}
}
//子函数模块代码 ADC和GPIO的初始化 以及 ADC读取结果
#include "stm32f10x.h" // Device header
#include "Delay.h"
void AD_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);
ADC_RegularChannelConfig(ADC1,ADC_Channel_0,1,ADC_SampleTime_28Cycles5);
//用结构体配置ADC
ADC_InitTypeDef ADC_InitStructure;
ADC_InitStructure.ADC_Mode=ADC_Mode_Independent;
ADC_InitStructure.ADC_DataAlign=ADC_DataAlign_Right;
ADC_InitStructure.ADC_ExternalTrigConv=ADC_ExternalTrigConv_None;
ADC_InitStructure.ADC_ContinuousConvMode=ENABLE;//指定扫描模式是连续扫描模式
ADC_InitStructure.ADC_ScanConvMode=DISABLE;
ADC_InitStructure.ADC_NbrOfChannel=1;
ADC_Init(ADC1,&ADC_InitStructure);
ADC_Cmd(ADC1,ENABLE);
ADC_ResetCalibration(ADC1);
while(ADC_GetResetCalibrationStatus(ADC1)==SET);
ADC_StartCalibration(ADC1);
while (ADC_GetCalibrationStatus(ADC1)==SET);
//因为是连续转换模式,所以只需要触发一次即可连续转换。
ADC_SoftwareStartConvCmd(ADC1,ENABLE);//软件触发ADC开始开始转换
}
uint16_t AD_GetValue(void)
{
return ADC_GetConversionValue(ADC1);
//连续转换,不需要判断标志位
//获取ADC_DR(DR数据寄存器)的值,不需要手动清除标志位了
}