基于STM32F103的ADC采集详解
创作时间:
作者:
@小白创作中心
基于STM32F103的ADC采集详解
引用
CSDN
1.
https://blog.csdn.net/m0_61313322/article/details/142774512
本文将详细介绍如何在STM32F103微控制器上实现ADC(模数转换)采集。内容包括ADC的初始化配置、通道配置、数据读取以及定时器中断的设置,适合对STM32开发感兴趣的工程师和爱好者参考。
ADC初始化配置
void Adc_Init(void)
{
ADC_InitTypeDef ADC_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOC|RCC_APB2Periph_ADC1, ENABLE ); //使能ADC1通道时钟
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M
//模拟通道输入引脚
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_4|GPIO_Pin_3|GPIO_Pin_2|GPIO_Pin_5|GPIO_Pin_6;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOA, &GPIO_InitStructure);
//GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
//GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
//GPIO_Init(GPIOB, &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚
GPIO_Init(GPIOC, &GPIO_InitStructure);
ADC_DeInit(ADC1); //复位ADC1
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式
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); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器
ADC_ITConfig(ADC1, ADC_IT_EOC, ENABLE);
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1
ADC_ResetCalibration(ADC1); //使能复位校准
while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束
ADC_StartCalibration(ADC1); //开启AD校准
while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能
//设置指定ADC的规则组通道,一个序列,采样时间
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_13Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期
ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能
}
获取ADC值
u8 aaa;
u16 Get_Adc(u8 ch)
{
//设置指定ADC的规则组通道,一个序列,采样时间
ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_13Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期

ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能
delay_us(2);
//while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束
return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果
}
void AD_config(u8 i)
{
AD_CY0[i] =Get_Adc(ADC_Channel_0);//PA0
AD_CY1[i] =Get_Adc(ADC_Channel_1);//PA1
AD_CY2[i] =Get_Adc(ADC_Channel_2);//PA2
AD_CY3[i] =Get_Adc(ADC_Channel_3);//PA3
AD_CY5[i] =Get_Adc(ADC_Channel_5);//PA4
AD_CY6[i] =Get_Adc(ADC_Channel_6);//PB1
AD_CY10[i] =Get_Adc(ADC_Channel_10);//PC0
AD_CY11[i] =Get_Adc(ADC_Channel_11);//PC1
AD_CY12[i] =Get_Adc(ADC_Channel_12);//PC2
AD_CY13[i] =Get_Adc(ADC_Channel_13);//PC3
}
u16 Get_Adc_Average(u8 ch,u8 times)
{
uint32_t temp_val=0;
u8 t;
for(t=0;t<times;t++)
{
temp_val+=Get_Adc(ch);
// Delay_ms();
}
return temp_val/(times/**4*/);
}
uint32_t temp_data(u32 wd)
{
u32 wd0=0,wd1=0,i2=0,wdbz=0;
while((wdbz==0)&&(i2<22))
{
if((bufad1[i2]<=wd)&&(wd<bufad1[i2+1]))
{
wd1 = bufwd1[i2];
ceshi4 = bufwd1[i2];
wd0=(wd-bufad1[i2])*10/(bufad1[i2+1]-bufad1[i2]);
wd0+=wd1;
wdbz=1;
}
i2++;
ceshi3 = i2;
}
if(wdbz==0){if(wd<36){wd0=0;}else if(wd >3373){wd0=200;}}
return wd0;
}
u32 AD_avg(u32 *adcy)
{
u32 adavg40=0,adavg41=0;
u8 i;
for(i=0;i<CYnumb;i++)
{
adavg40 +=adcy[i];
}
adavg40 /=CYnumb;
for(i=0;i<CYnumb;i++)
{
if(adcy[i]>adavg40){adavg41=adavg41+adcy[i]-adavg40;}
else{adavg41=adavg41-adcy[i]+adavg40;}
}
//adavg41 /=40;
return adavg41;
}
定时器中断配置
void TIM2_Int_Init(u16 arr,u16 psc)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; //定时器配置结构体定义
NVIC_InitTypeDef NVIC_InitStructure; //中断配置结构体定义
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); //时钟使能
//定时器TIM2初始化
TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx的时间基数单位
TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE ); //使能指定的TIM2中断,允许更新中断 TIM_IT_Update
//中断优先级NVIC设置
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn; //TIM2中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级0级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; //从优先级1级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
NVIC_Init(&NVIC_InitStructure); //初始化NVIC寄存器
TIM_Cmd(TIM2, ENABLE); //使能TIM2
}
u16 countadc;
void TIM2_IRQHandler(void) //TIM2中断
{
countadc++;
while (countadc==40)//AD采集
{
countadc=0;
AD_jisuan();
}
TIM_ClearITPendingBit(TIM2, TIM_IT_Update); // 清除中断标志
}
主函数初始化
在main函数中,需要初始化定时器和ADC:
int main(void)
{
// 初始化系统时钟
SystemInit();
// 初始化ADC
Adc_Init();
// 初始化定时器
TIM2_Int_Init(41999,71);
// 主循环
while(1)
{
// 主循环代码
}
}
通过以上步骤,可以实现基于STM32F103的ADC采集功能。定时器TIM2以0.5ms的间隔触发中断,进行ADC数据采集和处理。
热门推荐
创业公司 vs 成熟公司:求职者该如何选择?
如何选择达人,揭示筛选标准的秘密与挑战
北证50指数狂飙背后:估值修复、业绩支撑与市场风格切换共舞,后市投资潜力引关注
《哪吒2》凭何一骑绝尘?|南方深读
应对“顽固”血脂,新型降脂药PCSK9抑制剂新在哪?
夏季食用鱿鱼安全指南:如何辨别是否被甲醛浸泡?
撸串的烤鱿鱼有没有被甲醛泡过?夏季鱿鱼怎么选吃得安心?食品专家来支招
运用线上分时预约系统,有效控制景区内人流
各类茶叶的冲泡技巧与种类选择
成为律师后如何选择专业领域
法学专业的职业选择有哪些
基金收益率的计算方法及其对基金表现的反映
赤玉石与麦饭石铺面对比:全面分析多肉植物适用的铺面材料选择指南
走访千年古都河南洛阳 “穿越之旅”如何玩出新花样?
颠覆认知!重新解读“东岳三杰”
父权关系是什么?如何打破传统父权结构?
使用碳纤维模具相关零件的 3 个技巧
教你如何判断是否需要补牙,避免被牙医"忽悠"!
小朋友牙齿黑黑的刷不掉怎么办?是蛀牙吗?牙医话你知4大造成牙齿黑黑成因
四大金刚的司仪是什么
笑晕!怪不得苹果直营店员工上班很热情!原来待遇这么好 真是太羡慕
后视镜防雨膜贴膜过程详解
一天吃一个牛油果减肥还是增肥
牛油果吃了有什么好处
聚焦智慧民航|5G-A:加速构建“数字天空”
让头发“疯狂生长”,吃对这些就够了!
旧物改造衣服:废旧材料的时尚新生
时尚圈掀起了一股“环保风”,2025年春季可持续时尚穿搭趋势解析
鼻骨骨折的症状体征有哪些
加钢淬火,锻造新时代民兵——多地推进民兵训练基地转型升级提升应战应急能力见闻