【stm32】ADC的介绍与使用
ADC的介绍与使用
1、ADC介绍
ADC(Analog-Digital Converter)模拟-数字转换器
ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁
12位逐次逼近型ADC,1us转换时间
两个重要参数:
1MHZ为STM32 ADC的最快转换频率
输入电压范围:03.3V,转换结果范围:04095
18个输入通道,可测量16个外部和2个内部信号源
规则组和注入组两个转换单元
模拟看门狗自动监测输入电压范围
STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道
2、逐次逼近型ADC
3、ADC电路
4、ADC基本结构
程序代码编写:
步骤:
1、开启RCC时钟,包括ADC和GPIO时钟,ADCCLK的分频器配置
2、配置GPIO,把需要用的GPIO配置成模拟输入的模式
3、配置多路开关,把左边的通道接入到右边的规则组列表里
4、配置ADC转换器,使用库函数结构体配置
5、开关控制,调用ADC_Cmd函数,开启ADC
ADC 通道和引脚复用的关系
5、转换模式
(1)单次转换,非扫描模式
只转换一次,每次只有一个通道,读取EOC标志位的值为1时,读取结束,此时可读取ADC的值
转换流程:
1、软件触发转换
2、等待转换完成
3、读取ADC数据寄存器
程序编写:
// 单通道AD转换
// AD.c
#include "stm32f10x.h" // Device header
// 单次转换,非扫描模式
void AD_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 72M/6 = 12M
// 初始化PA0为模拟输入的引脚
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; // 模拟输入模式 在AIN模式下,GPIO口是无效的
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
// 在规则组菜单列表的第一个位置,写入通道0 , 55.5 + 12.5 = 68个ADC周期
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);// 选择规则组输入通道
ADC_InitTypeDef ADC_InitStructure;
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; // ADC 模式---独立模式
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // 数据对齐---右对齐
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;// 外部触发转换选择(触发控制的触发源)---软件触发
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; // 连续转换模式---单次转换
ADC_InitStructure.ADC_ScanConvMode = DISABLE; // 扫描转换模式---非扫描模式
ADC_InitStructure.ADC_NbrOfChannel = 1; // 通道数目
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE); // 开启ADC
// ADC校准
ADC_ResetCalibration(ADC1); // 复位校准
while (ADC_GetResetCalibrationStatus(ADC1) == SET); // 等待复位校准完成
ADC_StartCalibration(ADC1); // 开始校准
while (ADC_GetCalibrationStatus(ADC1) == SET); // 等待校准完成
}
// 启动转换,获取结果
uint16_t AD_GetValue(void)
{
ADC_SoftwareStartConvCmd(ADC1, ENABLE); // 软件触发转换
// 等待时间5.6us, (1/12M)*68个ADC周期 = 5.6us
while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);// 等待转换完成
return ADC_GetConversionValue(ADC1); // 读取ADC数据寄存器
}
// main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"
uint16_t ADValue;
float Voltage;
/*
AD采样值来回抖动解决方法:
1、迟滞比较方法:
如开关灯,设置两个阈值,低于下阈值时,开灯,高于上阈值时,关灯
2、滤波方法:均值滤波,
读10个或20个值,取平均值,作为滤波的AD值,
*/
//
int main(void)
{
OLED_Init();
AD_Init();
OLED_ShowString(1, 1, "ADValue:");
OLED_ShowString(2, 1, "Volatge:0.00V");
while (1)
{
ADValue = AD_GetValue();
Voltage = (float)ADValue / 4095 * 3.3; // AD值转换成电压
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);
}
}
(2)连续转换,非扫描模式
起始触发一次,后续可持续转换,需要ADC的值时直接读取即可
程序编写:
// 单通道AD转换
// AD.c
#include "stm32f10x.h" // Device header
// 连续转换,非扫描模式
void AD_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div6); // 72M/6 = 12M
// 初始化PA0为模拟输入的引脚
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; // 模拟输入模式 在AIN模式下,GPIO口是无效的
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
// 在规则组菜单列表的第一个位置,写入通道0 , 55.5 + 12.5 = 68个ADC周期
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);// 选择规则组输入通道
ADC_InitTypeDef ADC_InitStructure;
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; // ADC 模式---独立模式
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; // 数据对齐---右对齐
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;// 外部触发转换选择(触发控制的触发源)---软件触发
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; // 连续转换模式---单次转换
ADC_InitStructure.ADC_ScanConvMode = DISABLE; // 扫描转换模式---非扫描模式
ADC_InitStructure.ADC_NbrOfChannel = 1; // 通道数目
ADC_Init(ADC1, &ADC_InitStructure);
ADC_Cmd(ADC1, ENABLE); // 开启ADC
// ADC校准
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数据寄存器
}
// main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"
uint16_t ADValue;
float Voltage;
/*
AD采样值来回抖动解决方法:
1、迟滞比较方法:
如开关灯,设置两个阈值,低于下阈值时,开灯,高于上阈值时,关灯
2、滤波方法:均值滤波,
读10个或20个值,取平均值,作为滤波的AD值,
*/
//
int main(void)
{
OLED_Init();
AD_Init();
OLED_ShowString(1, 1, "ADValue:");
OLED_ShowString(2, 1, "Volatge:0.00V");
while (1)
{
ADValue = AD_GetValue();
Voltage = (float)ADValue / 4095 * 3.3; // AD值转换成电压
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);
}
}
(3)单次转换,扫描模式
触发一次,转换结束就停止,下次转换需要再触发才能开始,可一次转换7个通道,7个通道转换完成后,产生EOC信号,转换结束
ps: 使用单次转换,非扫描模式实现多通道转换:在每次触发转换之前,先指定一下通道,再启动转换
程序编写:
// AD.c
// 多通道AD转换
#include "stm32f10x.h" // Device header
void AD_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
RCC_ADCCLKConfig(RCC_PCLK2_Div6);
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
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 = DISABLE;
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);
}
// param:ADC_Channel 指定转换的通道
uint16_t AD_GetValue(uint8_t ADC_Channel)
{
ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5); // 改变ADC通道
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);
return ADC_GetConversionValue(ADC1);
}
// main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"
uint16_t AD0, AD1, AD2, AD3;
int main(void)
{
OLED_Init();
AD_Init();
OLED_ShowString(1, 1, "AD0:");
OLED_ShowString(2, 1, "AD1:");
OLED_ShowString(3, 1, "AD2:");
OLED_ShowString(4, 1, "AD3:");
while (1)
{
// 获取AD值之前先改变通道
AD0 = AD_GetValue(ADC_Channel_0); // 获取通道0的AD数据
AD1 = AD_GetValue(ADC_Channel_1); // 获取通道1的AD数据
AD2 = AD_GetValue(ADC_Channel_2); // 获取通道2的AD数据
AD3 = AD_GetValue(ADC_Channel_3); // 获取通道3的AD数据
OLED_ShowNum(1, 5, AD0, 4);
OLED_ShowNum(2, 5, AD1, 4);
OLED_ShowNum(3, 5, AD2, 4);
OLED_ShowNum(4, 5, AD3, 4);
Delay_ms(100);
}
}
(4)连续转换,扫描模式
有多个通道,一次转换完成继续下一次转换且可以转换多个通道
ps:配合DMA使用
6、数据对齐

7、转换时间
T(conv) = 采样时间 + 12.5个ADC周期
T(conv) = 1.5 + 12.5 = 14个ADC周期 =1μs
(1/14M)*14个ADC采样周期 = 1μs
8、ADC校准
校准过程为固定的,只需要在ADC初始化的最后,加几条代码即可
9、ADC使用的几种硬件电路
1、电位器产生可调电压的电路:一个电位器产生一个可调的电压,电位器的两个固定端,一端接3.3V,另一端接GND,中间的滑动端可输出一个0~3.3V可调的电压输出,滑动端接ADC的输入通道,向上滑动,电压增大,反之减小
2、传感器输出电压电路:N1可以是光敏电阻、热敏电阻、红外接收管、麦克风等,都可以等效为一个可变电阻,此时电阻值无法直接测量,可以通过和一个固定电阻串联分压,来得到一个反应电阻值电压的电路,传感器(N1)阻值变小时,下拉作用变强,输出端(PA1)电压就下降,传感器(N1)阻值变大时,下拉作用变弱,输出端(PA1)受上拉电阻的作用,电压就会升高,固定电阻(R1)一般可以选择和传感器阻值相近的电阻,这样可以得到一个位于中间电压区域比较好的输出,图2传感器和固定电阻的位置也可以交换,此时输出电压的极性相反,这就是这个分压方法来输出传感器阻值的电路。
3、电压转换电路:例如要测一个0-5V的VIN电压,但ADC只能接收0-3.3V的电压,可以使用如上的电压转换电路,使用电阻进行分压,R1+R2 = 50k,根据分压公式 ,中间端的电压=VIN / 50k x 33k
,最后得到的电压范围就是0~3.3V,即可进入ADC转换。
作者:秋风&萧瑟