基于STM32的ADC电压采集(DMA传输)

我最近在写stm32的adc电压采集过程中遇到了一些问题,这些问题或许对stm32初学者入手adc有所帮助。

 对adc采集到的数据进行均值滤波处理,新建一个数组ADC_ConvertedValue_1ms[NOFCHANEL]用于存储所采集的值,对该变量求平均值放置于平均值数组(ADC_ConvertedValue_Average[x])中,打印均值滤波处理后的值即可

注:平均值数组用完记得清零

注:串口在调试过程中要记得先检查硬件配置,对应串口的RX,TX是否连接上

adc采集的通道记得要和对应的IO一一对应(否则会以adc通道所对应的IO口采集数据),如:PC0对应ADC通道10,PC1对应ADC通道11

以下为均值滤波处理数据代码示例

    while (1)
    {
			for(uint16_t i = 0; i < NOFCHANEL; i++)
			{
				ADC_ConvertedValue_Average += ADC_ConvertedValue_1ms[i];
			}
			ADC_ConvertedValue_Average /= NOFCHANEL;
			 if ( time == 1000 ) /* 1000 * 1 ms = 1s 时间到 */
      {
      time = 0;
      ADC_Vol = (float) ADC_ConvertedValue_Average/4096*3.3; // 读取转换的AD值
      printf("\r\n The current AD value = %f V \r\n",ADC_Vol);
			
//				for(uint16_t i = 0; i < NOFCHANEL; i++)
//				{
//					printf("\r\n The current AD value = %f V \r\n",(float) ADC_ConvertedValue[i]/4096*3.3);
//				}
			}
			//平均值数值置零
			ADC_ConvertedValue_Average = 0;
		}

此外,除了均值滤波外,还有个更为方便的加权均值滤波,其基本计算公式为

//先赋初值给均值变量 ADC_value_Average
ADC_value_Average = ADC_value;

//其次将每次的采样值用以下公式计算(跟历史采样值有关,N代表采样次数)
// S(t) = (S(t - 1) * (N - 1) + C) / N
ADC_value_Average(t) = (ADC_value_Average(t - 1) * (N - 1) + ADC_value)/N;

/*该滤波器可与定时器中断配合来实现滤波的功能*/

函数:HAL_ADC_Start_DMA(&ADC_Handle, (uint32_t*)&ADC_ConvertedValue, NOFCHANEL);

第一个参数是ADC的操作句柄;第二个参数是用来保存ADC转换结果数组的首地址第三个参数是转换的数据长度,当设置字长后可通过其来设置采样次数(若为1,则只有首地址有ADC的采样值,若为数组长度,则采样数据可填赋满整个数组)。

  • 2
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是基于STM32CubeIDE的STM32F103C8T6 ADC电压采集程序范例,假设我们要采集PA0引脚上的电压。 首先,需要配置ADC通道和引脚,并启动ADC转换。ADC的输入电压范围应该在0V到3.3V之间,因此建议使用一个分压电路将输入电压缩小到该范围内,以避免超出范围。 ```c // 配置PA0为ADC输入 GPIO_InitTypeDef GPIO_InitStruct = {0}; __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitStruct.Pin = GPIO_PIN_0; GPIO_InitStruct.Mode = GPIO_MODE_ANALOG; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); // 配置ADC通道 ADC_ChannelConfTypeDef sConfig = {0}; __HAL_RCC_ADC1_CLK_ENABLE(); hadc1.Instance = ADC1; hadc1.Init.ScanConvMode = DISABLE; hadc1.Init.ContinuousConvMode = DISABLE; hadc1.Init.DiscontinuousConvMode = DISABLE; hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START; hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT; hadc1.Init.NbrOfConversion = 1; HAL_ADC_Init(&hadc1); sConfig.Channel = ADC_CHANNEL_0; sConfig.Rank = ADC_REGULAR_RANK_1; sConfig.SamplingTime = ADC_SAMPLETIME_13CYCLES_5; HAL_ADC_ConfigChannel(&hadc1, &sConfig); // 启动ADC转换 HAL_ADC_Start(&hadc1); ``` 然后,在主循环中等待ADC转换完成,并获取采样数据。将ADC输出值转换为电压值,并通过串口输出。 ```c while (1) { // 等待ADC转换完成 if (HAL_ADC_PollForConversion(&hadc1, 100) == HAL_OK) { // 读取采样数据 uint32_t adc_value = HAL_ADC_GetValue(&hadc1); // 将ADC输出值转换为电压值 float voltage = adc_value * 3.3 / 4096; // 输出采样电压值 char buffer[32]; sprintf(buffer, "Voltage: %.2fV\r\n", voltage); HAL_UART_Transmit(&huart1, (uint8_t *)buffer, strlen(buffer), HAL_MAX_DELAY); // 启动下一次ADC转换 HAL_ADC_Start(&hadc1); } } ``` 需要注意的是,ADC转换的精度和采样速率受到多个因素的影响,例如ADC时钟频率、采样时间、参考电压等。在实际应用中需要根据具体需求进行调整。同时,串口输出也需要根据实际情况进行配置。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值