系列文章目录
目录
前言
前面我们已经学习了PWM的功能与其在我们项目中是一个调节LED光强的角色,现在我们要学习的是ADC采样,有了它,我们才能实现我们的自动调节灯光亮度。
提示:以下是本篇文章正文内容,下面案例可供参考
一、ADC是什么?
ADC模块中文名为模拟/数字转换器,是12位逐次逼近型的模拟数字转换器,一般用于数值的采样。我们学习ADC采样需要知道三个重要参数:采样位数、采样率、采样精度。
1.采样位数
采样位数越高,能够分解的位数越多,我们就能测得更准。就像我们之前使用过的万用表去测电压,用红、黑表笔可以测出待测电路的电压,ADC采样也类似。它可以将待测点的电压模拟量转化成数字量,如图:
待测点 | ADC采样测量值 |
0.00V | 0 |
3.3V | 4095 |
而为何3.3V是4095呢?是因为我们的STM32F103单片机是使用12位采样位数ADC,12位的二进制数的话,它的范围只能是0~4095:
4095 = 1111 1111 1111 = 2的12次方 - 1
相当于我们将0~3.3V分成相等的4096份,这样的话,每份转换成电压模拟量就是:
现在如果我们ADC采集的测量量是0V,我们ADC采集返回的数值就是0;如果我们ADC采集的测量量是0.0008V,我们ADC采集返回的数值就是1;以此类推,返回的数值如果是10,那么我们就知道我们测量到的电压是0.008V。
但是,当这个待测点的电压为0.0004V时,我们去测量的ADC返回值是0,就和0V一样,我们只能看到ADC返回值,所以我们会以为待测点的电压为0V,实则不是!这个就是采样位数少带来的测量能力不同。
12位的ADC无法分辨0V与0.0004V,因为我们把0~3.3V分成了4096份,我们如果再把份数增多,就能测量的更加精细。我们就其分成8192份,会有什么情况:
现在每份是0.0004V,那我们就可以测量到0.0004V,ADC的返回值是1,而不是0,就可以区分啦。分成8192份的话,就是13位ADC。以此类推,我们的份数分的越多,我们的采样位数就越高,我们测得就越准、越精细。
2.采样率
如果用ADC去采集直流电压,比如一个稳定的1V。可是如果我们要采集的是一个在不断变化的信号呢?比如采集1hz正弦信号(一个周期为1s),我们如果每1s采集一次,那么完全看不出这个信号的变化和波形;但如果我们每100ms采集一次呢,一个信号周期内就采集了10个点,可以大致看出这个信号的波形;如果我们每1ms,每1us采集一次呢?那么这个信号的波形就完全被我们知道了。
上面的1s,100ms,1us分别对应的是1 sps、10 sps、1M sps采样率。采样率表示的是ADC的采样速度。对于不同ADC,有自己的采样率上限的。比如STM32F103的单个ADC采样率支持1hz-1Mhz。
拿1M采样率去采集1K信号,一个周期有1000个点,效果非常好;但是如果1M去采集1M信号,这和每1s去采集一个1hz信号一样,一个周期1个点,根本看不出波形。
3.采样精度
采样精度很容易和采集位数的概念混淆,我们使用0~3.3V的12位ADC测量一个0.08V的待测量,理论值应该是:
但是,我们实际的测量值会在95~105之间波动,ADC测量会有误差,12位ADC的每份对应的是0.0008V(0.8mv)的电压,但是它会有10mv的波动电压,因此我们12位ADC只能当作8-9位的理想ADC使用,8-9位就是ADC的采样精度。
二、库函数配置步骤
以下使用正点原子的STM32F103战舰开发板为例(参考正点原子官方步骤):
①开启 PA 口时钟和 ADC1 时钟,设置 PA1 为模拟输入。
②复位 ADC1,同时设置 ADC1 分频因子。
③初始化 ADC1 参数,设置 ADC1 的工作模式以及规则序列的相关信息。
④使能 ADC 并校准。
⑤读取 ADC 值。
1.时钟、引脚配置
时钟配置与引脚配置代码如下:
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE );
//使能ADC1通道时钟
//PA1 作为模拟通道输入引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOA, &GPIO_InitStructure);
2.复位ADC
RCC_ADCCLKConfig(RCC_PCLK2_Div6);
ADC_DeInit(ADC1);
3.初始化ADC1参数
void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);
typedef struct{uint32_t ADC_Mode;FunctionalState ADC_ScanConvMode;FunctionalState ADC_ContinuousConvMode;uint32_t ADC_ExternalTrigConv;uint32_t ADC_DataAlign;uint8_t ADC_NbrOfChannel;}ADC_InitTypeDef;
ADC_InitTypeDef ADC_InitStructure;
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC 工作模式:独立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //AD 单通道模式
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //AD 单次转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
//转换由软件而不是外部触发启动
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC 数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的 ADC 通道的数目 1
ADC_Init(ADC1, &ADC_InitStructure); //根据指定的参数初始化外设 ADCx
4.使能ADC并校准
ADC_Cmd(ADC1, ENABLE);//使能指定的 ADC1
ADC_ResetCalibration(ADC1);
ADC_StartCalibration(ADC1);//开始指定 ADC1 的校准状态
while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束while(ADC_GetCalibrationStatus(ADC1)); //等待校 AD 准结束
以上步骤组成了我们的ADC初始化Adc_Init(void),代码如下:
//初始化 ADC
//这里我们仅以规则通道为例
//我们默认将开启通道 0~3
void Adc_Init(void)
{ ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1 , ENABLE ); //使能 ADC1 通道时钟
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置 ADC 分频因子 6
//72M/6=12,ADC 最大时间不能超过 14M
//PA1 作为模拟通道输入引脚
GPIO_InitStructure.GPIO_Pin =GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;//模拟输入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化 GPIOA.1
ADC_DeInit(ADC1); //复位 ADC1,将外设 ADC1 的全部寄存器重设为缺省值
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC 独立模式
ADC_InitStructure.ADC_ScanConvMode = DISABLE; //单通道模式
ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //单次转换模式
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//转换由
//软件而不是外部触发启动
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC 数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 1; //顺序进行规则转换的ADC 通道的数目
ADC_Init(ADC1, &ADC_InitStructure); //根据指定的参数初始化外设 ADCx
ADC_Cmd(ADC1, ENABLE); //使能指定的 ADC1
ADC_ResetCalibration(ADC1); //开启复位校准
while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束
ADC_StartCalibration(ADC1); //开启 AD 校准
while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束
}
5. 读取ADC值
void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel,uint8_t Rank, uint8_t ADC_SampleTime);
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );
ADC_SoftwareStartConvCmd(ADC1, ENABLE);//使能指定的 ADC1 的软件转换启动功能
ADC_GetConversionValue(ADC1);
FlagStatus ADC_GetFlagStatus(ADC_TypeDef* ADCx, uint8_t ADC_FLAG)
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束
获取ADC值的函数 Get_Adc(u8 ch) 代码如下:
//获得 ADC 值
//ch:通道值 0~3
u16 Get_Adc(u8 ch)
{
//设置指定 ADC 的规则组通道,设置它们的转化顺序和采样时间
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );
//通道 1,规则采样顺序值为 1,采样时间为 239.5 周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能软件转换功能
while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束
return ADC_GetConversionValue(ADC1); //返回最近一次 ADC1 规则组的转换结果
}
三、调试过程
1.烧录代码
使用正点原子的官方例程,代码我已放入网盘:
链接:https://pan.baidu.com/s/1zMrL3K2HMrQHllIBsqJZbg?pwd=adc1
提取码:adc1
烧录后,现象:
ADC_CHO_VAL的值是ADC的返回值, ADC_CHO_VOL是根据公式换算后的模拟量,也就是电压值。
2.测量电压
我们使用一根杜邦线分别测量开发板上的3.3V引脚与GND引脚,观察ADC的返回值是否正确。(
- 使用杜邦线连接PA1与3.3V引脚,观察LCD,发现ADC返回值为4095,电压值为3.299V,约等于3.3V:
- 使用杜邦线连接PA1与GND引脚,观察LCD,发现ADC返回值为0,电压值为0.000V:
总结
本文学习了ADC是干什么用的,对于我们的智能灯光调节器项目,ADC采集的使用至关重要,项目中的自动模式将会对ADC进行使用,它也是单片机的一个很重要的功能,需要反复学习。后续将会有光敏传感器结合ADC采样的使用,去慢慢贴合我们的项目,一步步打造你的第一个STM32小项目!!!加油!!!