对于STM32内部ADC采集时候,难免会受到外部干扰的,导致数据不稳定情况。一般在实际使用过程中,我们就需要对增加采样次数来提高分辨率的技术。而STM32cube MX 软件自带过采样的功能。下面就和大家介绍一下STM32如何使用软件配置过采样的功能,从而实现数据的稳定性。
下面和大家具体分享一下软件配置步骤和注意事项:
一:过采样的基本知识分享:
1.1 基本的原理:使用STM32通过多次采取ADC的数据,并对结果进行取平均值处理,提高有效分辨率。每增加4倍的采样次数,分辨率可提高1位;我们在实际的使用过程中,可以根据自己的需求进行设置和更改。
二:软件配置步骤
2.1 配置ADC
选择ADC通道:确定要采样的模拟输入通道。
设置采样时间:根据信号特性调整采样时间。
配置ADC分辨率:通常设置为12位。
2.2 配置过采样
设置过采样比率:例如16倍过采样。
设置右移位数:根据过采样比率调整,如16倍过采样需右移2位。
2.3 启动ADC
启动转换:ADC开始采样并存储结果。
等待转换完成:通过轮询或中断方式检查转换状态。
2.4 读取和处理数据
读取ADC数据:从数据寄存器获取采样值。
计算平均值:对多次采样结果取平均,得到最终值。
STM32 cube MX 软件配置如下:

主要代码如下所示:
/* Start ADC group regular conversion */
if (HAL_ADC_Start(&hadc1) != HAL_OK)
{
/* ADC conversion start error */
Error_Handler();
}
/* Wait for ADC conversion completed */
if (HAL_ADC_PollForConversion(&hadc1, 10) != HAL_OK)
{
/* End Of Conversion flag not set on time */
Error_Handler();
} */
uhADCxConvertedData_OVS_ratio16_shift4 = HAL_ADC_GetValue(&hadc1);
HAL_ADC_Stop(&hadc1);
hadc1.Init.OversamplingMode = ENABLE;
hadc1.Init.Oversampling.Ratio = ADC_OVERSAMPLING_RATIO_16;
hadc1.Init.Oversampling.RightBitShift = ADC_RIGHTBITSHIFT_NONE;
if (HAL_ADC_Init(&hadc1) != HAL_OK)
{
Error_Handler();
}
/* Start ADC group regular conversion */
if (HAL_ADC_Start(&hadc1) != HAL_OK)
{
/* ADC conversion start error */
Error_Handler();
}
if (HAL_ADC_PollForConversion(&hadc1, 10) != HAL_OK)
{
/* End Of Conversion flag not set on time */
Error_Handler();
}
uhADCxConvertedData_OVS_ratio16_shift0 = HAL_ADC_GetValue(&hadc1);
HAL_ADC_Stop(&hadc1);
hadc1.Init.OversamplingMode = DISABLE;
if (HAL_ADC_Init(&hadc1) != HAL_OK)
{
Error_Handler();
}
if (HAL_ADC_Start(&hadc1) != HAL_OK)
{
/* ADC conversion start error */
Error_Handler();
}
if (HAL_ADC_PollForConversion(&hadc1, 10) != HAL_OK)
{
/* End Of Conversion flag not set on time */
Error_Handler();
} */
uhADCxConvertedData_OVS_disabled = HAL_ADC_GetValue(&hadc1);
HAL_ADC_Stop(&hadc1);
hadc1.Init.OversamplingMode = ENABLE;
hadc1.Init.Oversampling.Ratio = ADC_OVERSAMPLING_RATIO_16;
hadc1.Init.Oversampling.RightBitShift = ADC_RIGHTBITSHIFT_4;
if (HAL_ADC_Init(&hadc1) != HAL_OK)
{
Error_Handler();
}
/* Data value should not exceed range resolution 12 bits */
if ((uhADCxConvertedData_OVS_ratio16_shift4 > __LL_ADC_DIGITAL_SCALE(LL_ADC_RESOLUTION_12B)) ||
(uhADCxConvertedData_OVS_disabled > __LL_ADC_DIGITAL_SCALE(LL_ADC_RESOLUTION_12B)))
{
/* Error: Data not valid */
Error_Handler();
}
/* Data value should not exceed range of resolution 16 bits */
if (uhADCxConvertedData_OVS_ratio16_shift0 > 0xFFFF)
{
/* Error: Data not valid */
Error_Handler();
}
if (fConvertedData_OVS_EquivalentValue12bits > __LL_ADC_DIGITAL_SCALE(LL_ADC_RESOLUTION_12B))
{
/* Error: Data not valid */
Error_Handler();
}ADC 配置为在单转换模式下从 SW 触发器转换单通道。ADC 过采样功能与 3 个设置一起使用:
第一步:启用 ADC 过采样:比率 16,位右移 4。
第二步:启用 ADC 过采样:比率 16,无位元右移。
第三步:禁用 ADC 过采样。
程序循环执行,上述步骤代码;
示例执行:
在主程序执行中,ADC 组使用 3 个过采样设置定期连续转换所选通道。
然后,对数据进行评估:
数据范围有效性检查
用户可以评估过采样的预期结果:启用过采样的 ADC 转换数据比禁用过采样的 ADC 转换数据的变化更小。
对 16 位的过采样转换数据进行软件计算,以获得相当于浮点分辨率 12 位的精确数据。
程序仿真效果如下所示:

注意事项:
噪声水平:过采样适用于噪声较大的信号。
采样时间:确保采样时间足够捕获信号。
时钟频率:ADC时钟频率需满足过采样要求。
项目总结:
STM32 ADC过采样通过增加采样次数提高分辨率,适用于需要高精度但硬件分辨率不足的场景。合理配置过采样比率和右移位数是关键。
几种常用得滤波算法分享:
1:一阶互补滤波算法:
取值:k = 0-1,本次取值滤波结果 =(1-K),本次采样值需要加上上次滤波得结果数据
代码如下所示:
int firstdatadeal(int CurrValue, int lastValue, float K)
{ return K * CurrValue + (1-K) * lastValue;}从代码中,我们可以看到该种滤波算法还是有很大得弊端的,效果比较一般,对于高精度的场合不建议使用。
2: 采用中值算法滤波
主要是程序在执行的时候,连续采集N次(需要注意下这里下,这里的N必须取值奇数),程序需要按大到小或者从小到达的顺序进行排序,然后取中间数值做为有效值。
代码如下:
int MidValueDeal(int N)
{
int value_buf[N]; int i,j,k,temp;
for( i = 0; i < N; ++i)
{
value_buf[i] = HAL_ADC_GetValue(&hadc1);
}
for(j = 0 ; j < N-1; ++j)
{
for(k = 0; k < N-j-1; ++k)
{ //从小到大排序,冒泡法排序
if(value_buf[k] > value_buf[k+1])
{
temp = value_buf[k];
value_buf[k] = value_buf[k+1];
value_buf[k+1] = temp;
}
}
}
return value_buf[(N-1)/2];
}可以有效的去除数据因为外界干扰引起的数据的波动,对一些大滞后系统:比如温度、液位等变化缓慢的有良好的滤波效果,上述代码可以有效消除异常数据和平稳变化的采样值效果比较好;
3:算术平均数滤波
连续取值N个数据,对所有的数据进行取平均值;
代码如下:
int AverValueDeal(int N)
{
int sum = 0;
unsinged short i;
for(i = 0; i < N; ++i)
{
sum += HAL_ADC_GetValue(&hadc1);
}
return sum/N;
}这里取值时候,需要根据被控对象进行选择,N值取值过大,会导致响应过慢,数据的灵敏度过低,所以采用算术平均值滤波时候,需要我们格外的注意;
4:滑动滤波算法
把连续取N个采样值看成一个队列,队列的长度固定为N。每次采样到一个新数据放入队尾,并扔掉原来队首的一次数据(先进先出原则)。把队列中的N个数据进行算术平均运算,就可获得新的滤波结果;
#define N 10
int value_buf[N];
int sum=0; //数据总和
int Num=0; //当前队列中数组下标
int movedataFilter()
{
if(Num < N)
{
value_buf[Num] = HAL_ADC_GetValue(&hadc1);
sum += value_buf[Num];
Num++;
return sum/Num;
}
else
{
sum -= sum/N;
sum += HAL_ADC_GetValue(&hadc1);
return sum/N;
}
}好了,几种常用的滤波算法,已经和大家分享完毕,欢迎大家相互讨论。
我要赚赏金
