STM32入门笔记(03): ADC低通滤波(IIR)(SPL库函数版)(2)

news2025/1/11 13:01:38

A/D转换的常用技术有逐次逼近式双积分式并行式跟踪比较式等。目前用的较多的是前3种。

A/D转换器的主要技术指标

转换时间
分辨率

例如,8位A/D转换器的数字输出量的变化范围为0~255,当输入电压的满刻度为5V时,数字量每变化一个数字所对应输入模拟电压的值为5V/255=19.6mV,其分辨能力为19.6mV

转换精度

转换精度指的是转换后所得的结果相对于实际值的准确度,可以用满量程的百分比这一相对误差来表示,如±0.05%

ADC应用设计深入讨论尽管STM32内部集成了12位ADC,但在实际应用中,要想真正实现12位精度且比较稳定的ADC并不简单,需要进一步从硬件、软件方面进行综合、细致地考虑。下面介绍一些在ADC应用设计中应该考虑的几个要点:

工作电压的稳定性

AVCC是提供给ADC工作的电源,如果AVCC不稳定,就会影响ADC的转换精度

参考电压的确定

ADC的参考电压应稍大于输入电压的最高值。ADC的参考电压VREF可以选择为AVCC,或外接参考电压源,外接的参考电压源应该稳定。

采样时钟的选择

ADC时钟频率最大为14MHz。如果STM32系统时钟频率为56MHz时,一般为4分频,ADC时钟频率为14MHz,如果系统时钟频率为72MHz时,一般为6分频,ADC时钟频率为12MHz。

ADC通道的输入信号频率带宽取决于ADC时钟频率。把ADC通道配置为55.5个周期,若ADCCLK的时钟频率配置为12MHz,则ADC采样的时间计算公式如下。Tcovn=采样时间+12.5个周期

其中:Tcovn为总转换时间,采样时间是根据每个通道的ADC_SampleTime的值来决定的。后面的12.5个周期是ADC转换时量化所需要的

固定的周期,ADC的一次转换所需要的时间是Tcovn=(55.5+12.5)×(1/12),大约是5.67μs。STM32的ADC输入阻抗典型值为50kΩ,为了保证测量准确,被测信号源的输出阻抗要尽可能低。

模拟噪声的抑制

器件外部和内部的数字电路会产生电磁干扰,并会影响模拟测量的精度。如果对ADC的转换精度要求很高,则可以采用以下的技术来降低噪声的影响。

使模拟信号的通路尽可能地短,模拟信号连线应从模拟地的布线盘上通过,并使它们尽可能远离高速开关数字信号线。

STM32的AVCC引脚应该通过LC网络与数字端电源VCC相连。

如果某些ADC引脚作为通用数字输出口,那么在ADC转换过程中,不要改变这些引脚的状态。

校准


ADC - 电压

要将 ADC (模数转换器) 的原始数值转换为电压值,通常需要进行以下步骤:

  1. 确定 ADC 的分辨率
    例如,对于 STM32F103C8T6,ADC 的分辨率是 12 位,因此其数值范围是 0 到 4095。

  2. 确定参考电压
    确定 ADC 的参考电压 Vref。在 STM32F103C8T6 中,默认情况下Vref 通常是 3.3V,但这取决于具体的硬件配置。

  3. 读取 ADC 值
    读取 ADC 转换后的数值,例如 ADC_value 。

  4. 使用公式转换为电压值
    使用公式计算实际的电压值:
    在这里插入图片描述

    其中 ( N ) 是 ADC 的分辨率。例如,对于 12 位 ADC,( N = 12 )。

具体步骤如下:

  1. 读取 ADC 值:

    uint16_t ADC_value = HAL_ADC_GetValue(&hadc1);
    
  2. 计算电压值:

    float V_REF = 3.3; // 假设参考电压为 3.3V
    float voltage = (ADC_value / 4095.0) * V_REF;
    

以下是一个完整的例子,假设你使用 STM32 HAL 库:

#include "main.h"

ADC_HandleTypeDef hadc1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_ADC1_Init();

  HAL_ADC_Start(&hadc1);

  while (1)
  {
    HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY);
    uint16_t ADC_value = HAL_ADC_GetValue(&hadc1);
    float V_REF = 3.3;
    float voltage = (ADC_value / 4095.0) * V_REF;
    
    // 在这里使用计算出的电压值
  }
}

static void MX_ADC1_Init(void)
{
  ADC_ChannelConfTypeDef sConfig = {0};

  hadc1.Instance = ADC1;
  hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
  hadc1.Init.ContinuousConvMode = ENABLE;
  hadc1.Init.DiscontinuousConvMode = DISABLE;
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  hadc1.Init.NbrOfConversion = 1;
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }

  sConfig.Channel = ADC_CHANNEL_1;
  sConfig.Rank = ADC_REGULAR_RANK_1;
  sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
}

static void MX_GPIO_Init(void)
{
  __HAL_RCC_GPIOA_CLK_ENABLE();
}

void SystemClock_Config(void)
{
  // 时钟配置代码
}

这个例子展示了如何初始化 ADC1 并读取通道 1 (PA1) 的 ADC 值,然后将其转换为电压值。


如果将100V电压经过串联电阻分压,电阻R1为66K,R2为3K,这样如何通过ADC采样计算出电压?

将 100V 电压经过分压电路并通过 ADC 进行采样,计算出实际电压的方法如下:

  1. 分压电路公式
    分压电路的输出电压 Vout 可以用以下公式计算:
    在这里插入图片描述

    其中 Vin 是输入电压, R1 和 R2 是分压电阻。

  2. 具体分压电路参数

    在这里插入图片描述

    注意:在实际应用中,必须确保 ( V_{out} ) 不超过 ADC 的参考电压(例如 3.3V)。在这种情况下,分压比使得 Vout 超过了 ADC 允许的最大电压,需要调整分压电阻以确保 Vout 在安全范围内

  3. 假设安全分压电阻

在这里插入图片描述

  1. 通过 ADC 读取分压后的电压
    设定 ADC 的参考电压 ( V_{REF} = 3.3V )。
假设 ADC 读取的数值为 ( ADC_value ),分辨率为 12 位 (0 到 4095)。

  2. 计算实际电压

在这里插入图片描述

  1. 反向计算原始输入电压

在这里插入图片描述

假设使用 STM32 和 HAL 库,代码示例如下:

#include "main.h"

ADC_HandleTypeDef hadc1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_ADC1_Init();

  HAL_ADC_Start(&hadc1);

  while (1)
  {
    HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY);
    uint16_t ADC_value = HAL_ADC_GetValue(&hadc1);

    float V_REF = 3.3;
    float V_out = (ADC_value / 4095.0) * V_REF;

    float R1 = 96000.0; // 96KΩ
    float R2 = 3000.0;  // 3KΩ

    float V_in = V_out * (R1 + R2) / R2;

    // 使用计算出的 V_in
  }
}

static void MX_ADC1_Init(void)
{
  ADC_ChannelConfTypeDef sConfig = {0};

  hadc1.Instance = ADC1;
  hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
  hadc1.Init.ContinuousConvMode = ENABLE;
  hadc1.Init.DiscontinuousConvMode = DISABLE;
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  hadc1.Init.NbrOfConversion = 1;
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }

  sConfig.Channel = ADC_CHANNEL_1;
  sConfig.Rank = ADC_REGULAR_RANK_1;
  sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
}

static void MX_GPIO_Init(void)
{
  __HAL_RCC_GPIOA_CLK_ENABLE();
}

void SystemClock_Config(void)
{
  // 时钟配置代码
}

这个代码展示了如何读取 ADC 值并将其转换为实际电压,再通过分压公式计算出原始输入电压。注意在实际电路中,请确保所有电压和电阻值在安全范围内,以防止损坏设备。


ADC 软件滤波

ADC 读取到的值有抖动是常见现象,可以通过以下几种方法来进行滤波:

  1. 平均值滤波:读取多次 ADC 值,取平均值,可以平滑抖动。
  2. 中值滤波:读取多次 ADC 值,取中间值,可以去除突发噪声。
  3. 低通滤波器:通过数字滤波算法,平滑高频噪声。

以下是每种方法的具体实现:

1. 平均值滤波

#define NUM_SAMPLES 10

uint16_t Read_ADC_Averaged(ADC_HandleTypeDef* hadc)
{
    uint32_t sum = 0;
    for (int i = 0; i < NUM_SAMPLES; i++)
    {
        HAL_ADC_Start(hadc);
        HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
        sum += HAL_ADC_GetValue(hadc);
    }
    return sum / NUM_SAMPLES;
}

2. 中值滤波

#define NUM_SAMPLES 9

uint16_t Read_ADC_Median(ADC_HandleTypeDef* hadc)
{
    uint16_t samples[NUM_SAMPLES];
    for (int i = 0; i < NUM_SAMPLES; i++)
    {
        HAL_ADC_Start(hadc);
        HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
        samples[i] = HAL_ADC_GetValue(hadc);
    }
    // 排序
    for (int i = 0; i < NUM_SAMPLES - 1; i++)
    {
        for (int j = 0; j < NUM_SAMPLES - i - 1; j++)
        {
            if (samples[j] > samples[j + 1])
            {
                uint16_t temp = samples[j];
                samples[j] = samples[j + 1];
                samples[j + 1] = temp;
            }
        }
    }
    return samples[NUM_SAMPLES / 2];
}

3. 低通滤波器 (IIR 滤波器)

#define ALPHA 0.1 // 滤波系数,取值范围为 0 到 1

float Read_ADC_LowPass(ADC_HandleTypeDef* hadc, float previous_filtered_value)
{
    HAL_ADC_Start(hadc);
    HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
    uint16_t raw_value = HAL_ADC_GetValue(hadc);
    float current_filtered_value = ALPHA * raw_value + (1 - ALPHA) * previous_filtered_value;
    return current_filtered_value;
}

综合代码示例

将这些滤波方法整合到你的代码中,可以如下:

#include "main.h"

ADC_HandleTypeDef hadc1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);

#define NUM_SAMPLES 10
#define ALPHA 0.1

float previous_filtered_value = 0;

uint16_t Read_ADC_Averaged(ADC_HandleTypeDef* hadc)
{
    uint32_t sum = 0;
    for (int i = 0; i < NUM_SAMPLES; i++)
    {
        HAL_ADC_Start(hadc);
        HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
        sum += HAL_ADC_GetValue(hadc);
    }
    return sum / NUM_SAMPLES;
}

uint16_t Read_ADC_Median(ADC_HandleTypeDef* hadc)
{
    uint16_t samples[NUM_SAMPLES];
    for (int i = 0; i < NUM_SAMPLES; i++)
    {
        HAL_ADC_Start(hadc);
        HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
        samples[i] = HAL_ADC_GetValue(hadc);
    }
    for (int i = 0; i < NUM_SAMPLES - 1; i++)
    {
        for (int j = 0; j < NUM_SAMPLES - i - 1; j++)
        {
            if (samples[j] > samples[j + 1])
            {
                uint16_t temp = samples[j];
                samples[j] = samples[j + 1];
                samples[j + 1] = temp;
            }
        }
    }
    return samples[NUM_SAMPLES / 2];
}

float Read_ADC_LowPass(ADC_HandleTypeDef* hadc, float previous_filtered_value)
{
    HAL_ADC_Start(hadc);
    HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
    uint16_t raw_value = HAL_ADC_GetValue(hadc);
    float current_filtered_value = ALPHA * raw_value + (1 - ALPHA) * previous_filtered_value;
    return current_filtered_value;
}

int main(void)
{
    HAL_Init();
    SystemClock_Config();
    MX_GPIO_Init();
    MX_ADC1_Init();

    HAL_ADC_Start(&hadc1);

    while (1)
    {
        uint16_t ADC_value_averaged = Read_ADC_Averaged(&hadc1);
        uint16_t ADC_value_median = Read_ADC_Median(&hadc1);
        previous_filtered_value = Read_ADC_LowPass(&hadc1, previous_filtered_value);

        float V_REF = 3.3;
        float V_out_averaged = (ADC_value_averaged / 4095.0) * V_REF;
        float V_out_median = (ADC_value_median / 4095.0) * V_REF;
        float V_out_lowpass = (previous_filtered_value / 4095.0) * V_REF;

        float R1 = 96000.0;
        float R2 = 3000.0;

        float V_in_averaged = V_out_averaged * (R1 + R2) / R2;
        float V_in_median = V_out_median * (R1 + R2) / R2;
        float V_in_lowpass = V_out_lowpass * (R1 + R2) / R2;

        // 使用计算出的 V_in
    }
}

static void MX_ADC1_Init(void)
{
    ADC_ChannelConfTypeDef sConfig = {0};

    hadc1.Instance = ADC1;
    hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
    hadc1.Init.ContinuousConvMode = ENABLE;
    hadc1.Init.DiscontinuousConvMode = DISABLE;
    hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
    hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
    hadc1.Init.NbrOfConversion = 1;
    if (HAL_ADC_Init(&hadc1) != HAL_OK)
    {
        Error_Handler();
    }

    sConfig.Channel = ADC_CHANNEL_1;
    sConfig.Rank = ADC_REGULAR_RANK_1;
    sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
    if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
    {
        Error_Handler();
    }
}

static void MX_GPIO_Init(void)
{
    __HAL_RCC_GPIOA_CLK_ENABLE();
}

void SystemClock_Config(void)
{
    // 时钟配置代码
}

这个代码示例展示了如何分别使用平均值滤波、中值滤波和低通滤波来处理 ADC 抖动。根据实际需求选择合适的滤波方法,可以有效减小 ADC 读数的抖动。


低通滤波器(IIR)

低通滤波器是一种常用的信号处理技术,用于平滑信号并去除高频噪声。最常见的低通滤波器之一是指数加权移动平均滤波器(IIR 低通滤波器)。

原理

指数加权移动平均滤波器的基本原理是对当前采样值和先前的滤波值进行加权平均。公式如下:

在这里插入图片描述

具体实现步骤

  1. 定义变量

    • 滤波系数 (\alpha)
    • 上一次的滤波输出值(初始为 0)
  2. 读取 ADC 值

    • 使用 HAL 库读取 ADC 值
  3. 计算滤波值

    • 应用低通滤波公式
  4. 重复进行

    • 在循环中持续读取 ADC 值并计算滤波值

代码实现

以下是一个详细的 C 语言代码示例,使用 STM32 HAL 库实现低通滤波器:


uint16_t ADValue;			//定义AD值变量
float Voltage;				//定义电压变量
float Vin;          // 输入电压
#define ALPHA 0.1  // 滤波系数,范围在 0 到 1 之间

static float previous_filtered_value = 0;  // 上一次的滤波输出值

float Read_ADC_LowPass(float previous_filtered_value)
{
    ADValue = AD_GetValue();					//获取AD转换的值
    float current_filtered_value = ALPHA *(float)ADValue + (1 - ALPHA) * previous_filtered_value;
    return current_filtered_value;
}

int main(void)
{
    AD_Init();				//AD初始化

    while (1) {
		
        previous_filtered_value = Read_ADC_LowPass(previous_filtered_value);
        Voltage = (float)previous_filtered_value / 4095 * 3.3;		// 公式:adc_value/4096=Vout/Vref ,Vref=3.3
        // R1=10k R2=1k
        Vin = Voltage * 11.0; // 分压公式:Vout=Vin*(R2/(R1+R2)) 所以 Vin=Vout*((R1+R2)/R2)

        Delay_ms(100);			//延时100ms,手动增加一些转换的间隔时间
    }
}

详细说明

  1. 定义滤波系数和变量

    • #define ALPHA 0.1 定义滤波系数,决定新输入值的权重。
    • float previous_filtered_value = 0; 初始化上一次的滤波输出值。
  2. 读取 ADC 值并应用低通滤波公式

    • ADValue = AD_GetValue(); 获取 ADC 读取值。
    • float current_filtered_value = ALPHA *(float)ADValue + (1 - ALPHA) * previous_filtered_value; 应用低通滤波公式计算当前滤波值。
  3. 在主循环中持续滤波

    • previous_filtered_value = Read_ADC_LowPass(previous_filtered_value); 调用滤波函数,更新滤波值。
  4. 计算和使用滤波后的电压值

    • Voltage = (float)previous_filtered_value / 4095 * 3.3; 计算分压电路的输出电压。
    • Vin = Voltage * 11.0; 反向计算原始输入电压 Vin。

通过这个实现,可以有效地平滑 ADC 读取值的抖动,提高测量的稳定性。

在这里插入图片描述

在这里插入图片描述

程序下载

  • 测试程序:STM32_AD单通道_低通滤波IIR计算电压(20240704)

ADC - 电流

为了通过 ADC 采集电流,并根据 ADC 值计算出电流值,需要以下几个步骤:

  1. 选择电流传感器:通常使用分流电阻 (shunt resistor) 或霍尔效应电流传感器。
  2. 采样电压:通过分流电阻或传感器将电流转换为电压,然后用 ADC 采样该电压。
  3. 计算电流:根据电压值和已知的分流电阻或传感器的特性,计算出实际电流。

以下是详细步骤:

1. 分流电阻测量电流

使用分流电阻测量电流的方法如下:

在这里插入图片描述

2. ADC 采样电压

使用 ADC 采样分流电阻上的电压。假设 ADC 的参考电压 ( V_{REF} ) 为 3.3V,分辨率为 12 位 (0 到 4095)。

3. 计算电流

根据 ADC 采样到的电压值,计算实际电流:
[
I = \frac{V_{shunt}}{R_{shunt}} = \frac{ADC_{value} \times V_{REF}}{4095 \times R_{shunt}}
]

代码实现

以下是一个详细的 C 语言代码示例,使用 STM32 HAL 库通过分流电阻测量电流:

#include "main.h"

ADC_HandleTypeDef hadc1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);

#define R_SHUNT 0.01  // 分流电阻值 (单位: 欧姆)
#define V_REF 3.3     // ADC 参考电压 (单位: 伏特)
#define ADC_RESOLUTION 4095.0 // ADC 分辨率 (12 位)

float Read_Current(ADC_HandleTypeDef* hadc, float R_shunt)
{
    HAL_ADC_Start(hadc);
    HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
    uint16_t ADC_value = HAL_ADC_GetValue(hadc);
    float V_shunt = (ADC_value / ADC_RESOLUTION) * V_REF;
    float current = V_shunt / R_shunt;
    return current;
}

int main(void)
{
    HAL_Init();
    SystemClock_Config();
    MX_GPIO_Init();
    MX_ADC1_Init();

    HAL_ADC_Start(&hadc1);

    while (1)
    {
        float current = Read_Current(&hadc1, R_SHUNT);

        // 在这里使用计算出的电流值 current
    }
}

static void MX_ADC1_Init(void)
{
    ADC_ChannelConfTypeDef sConfig = {0};

    hadc1.Instance = ADC1;
    hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
    hadc1.Init.ContinuousConvMode = ENABLE;
    hadc1.Init.DiscontinuousConvMode = DISABLE;
    hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
    hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
    hadc1.Init.NbrOfConversion = 1;
    if (HAL_ADC_Init(&hadc1) != HAL_OK)
    {
        Error_Handler();
    }

    sConfig.Channel = ADC_CHANNEL_1;
    sConfig.Rank = ADC_REGULAR_RANK_1;
    sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
    if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
    {
        Error_Handler();
    }
}

static void MX_GPIO_Init(void)
{
    __HAL_RCC_GPIOA_CLK_ENABLE();
}

void SystemClock_Config(void)
{
    // 时钟配置代码
}

详细说明

  1. 定义分流电阻和 ADC 参数

    • #define R_SHUNT 0.01 定义分流电阻值,单位为欧姆。
    • #define V_REF 3.3 定义 ADC 参考电压,单位为伏特。
    • #define ADC_RESOLUTION 4095.0 定义 ADC 的分辨率。
  2. 读取 ADC 值并计算电流

    • HAL_ADC_Start(hadc); 启动 ADC。
    • HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY); 等待 ADC 转换完成。
    • uint16_t ADC_value = HAL_ADC_GetValue(hadc); 获取 ADC 读取值。
    • float V_shunt = (ADC_value / ADC_RESOLUTION) * V_REF; 计算分流电阻上的电压。
    • float current = V_shunt / R_shunt; 计算实际电流。
  3. 在主循环中持续测量电流

    • float current = Read_Current(&hadc1, R_SHUNT); 调用测量电流的函数,获取当前电流值。

通过这个实现,可以有效地使用 ADC 采集电流,并根据 ADC 值计算出实际电流值。


ADC - 温度

使用查表法计算温度是一种常见且简单的方法,特别适合在微控制器上实现,因为它不需要复杂的数学运算。以下是具体步骤和实现方法:

步骤概述

  1. 建立查表:根据 NTC 热敏电阻的特性曲线,创建一个电阻值与温度对应的查表(数组)。
  2. 读取 ADC 值:使用 ADC 读取分压电路的电压。
  3. 计算 NTC 电阻值:根据分压公式计算 NTC 热敏电阻的电阻值。
  4. 查找温度:在查表中找到对应的温度值。

具体步骤

1. 建立查表

NTC 热敏电阻的电阻值随温度变化。可以根据热敏电阻的数据手册或特性曲线建立一个电阻值与温度对应的查表。例如:

const uint32_t resistance_table[] = {3380, 3300, 3220, /* ... */ 100}; // 阻值 (单位: 欧姆)
const float temperature_table[] = {0, 1, 2, /* ... */ 100}; // 温度值 (单位: 摄氏度)
const int table_size = sizeof(resistance_table) / sizeof(resistance_table[0]);
2. 使用 ADC 读取电压

使用 STM32 的 ADC 采样 Vout。

3. 计算 NTC 电阻值

在这里插入图片描述

4. 查找温度

在查表中找到对应的温度值,可以使用线性插值法提高精度。

代码实现

以下是一个详细的 C 语言代码示例,使用 STM32 HAL 库实现以上步骤:

#include "main.h"
#include <stdio.h>

ADC_HandleTypeDef hadc1;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_ADC1_Init(void);

#define R1 10000  // 固定电阻值 (单位: 欧姆)
#define V_REF 3.3 // ADC 参考电压 (单位: 伏特)
#define ADC_RESOLUTION 4095.0 // ADC 分辨率 (12 位)

// 查表 (电阻值单位: 欧姆, 温度单位: 摄氏度)
const uint32_t resistance_table[] = {3380, 3300, 3220, /* ... */ 100};
const float temperature_table[] = {0, 1, 2, /* ... */ 100};
const int table_size = sizeof(resistance_table) / sizeof(resistance_table[0]);

float Read_NTC_Temperature(ADC_HandleTypeDef* hadc)
{
    HAL_ADC_Start(hadc);
    HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY);
    uint16_t ADC_value = HAL_ADC_GetValue(hadc);
    
    float Vout = (ADC_value / ADC_RESOLUTION) * V_REF;
    float R_ntc = R1 * (V_REF / Vout - 1);

    // 查表法找到温度值
    for (int i = 0; i < table_size - 1; i++)
    {
        if (R_ntc >= resistance_table[i+1] && R_ntc <= resistance_table[i])
        {
            // 线性插值计算温度
            float R1 = resistance_table[i];
            float R2 = resistance_table[i+1];
            float T1 = temperature_table[i];
            float T2 = temperature_table[i+1];
            float temperature = T1 + (R_ntc - R1) * (T2 - T1) / (R2 - R1);
            return temperature;
        }
    }

    // 如果未找到匹配的电阻值,返回一个默认值
    return -273.15; // 表示错误
}

int main(void)
{
    HAL_Init();
    SystemClock_Config();
    MX_GPIO_Init();
    MX_ADC1_Init();

    HAL_ADC_Start(&hadc1);

    while (1)
    {
        float temperature = Read_NTC_Temperature(&hadc1);

        // 使用计算出的温度值 temperature
    }
}

static void MX_ADC1_Init(void)
{
    ADC_ChannelConfTypeDef sConfig = {0};

    hadc1.Instance = ADC1;
    hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
    hadc1.Init.ContinuousConvMode = ENABLE;
    hadc1.Init.DiscontinuousConvMode = DISABLE;
    hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
    hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
    hadc1.Init.NbrOfConversion = 1;
    if (HAL_ADC_Init(&hadc1) != HAL_OK)
    {
        Error_Handler();
    }

    sConfig.Channel = ADC_CHANNEL_1;
    sConfig.Rank = ADC_REGULAR_RANK_1;
    sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
    if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
    {
        Error_Handler();
    }
}

static void MX_GPIO_Init(void)
{
    __HAL_RCC_GPIOA_CLK_ENABLE();
}

void SystemClock_Config(void)
{
    // 时钟配置代码
}

详细说明

  1. 定义常量和查表

    • #define R1 10000 定义固定电阻 ( R1 ) 的值,单位为欧姆。
    • #define V_REF 3.3 定义 ADC 参考电压,单位为伏特。
    • #define ADC_RESOLUTION 4095.0 定义 ADC 的分辨率。
    • const uint32_t resistance_table[]const float temperature_table[] 分别定义电阻值和温度值的查表。
    • const int table_size 定义查表的大小。
  2. 读取 ADC 值并计算电阻和温度

    • HAL_ADC_Start(hadc); 启动 ADC。
    • HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY); 等待 ADC 转换完成。
    • uint16_t ADC_value = HAL_ADC_GetValue(hadc); 获取 ADC 读取值。
    • float Vout = (ADC_value / ADC_RESOLUTION) * V_REF; 计算分压电路的输出电压。
    • float R_ntc = R1 * (V_REF / Vout - 1); 计算 NTC 热敏电阻的电阻值。
  3. 查找温度值

    • 遍历查表,通过线性插值计算电阻值与温度值之间的关系,找到对应的温度。

通过这个实现,可以有效地使用 ADC 采集 NTC 热敏电阻的电阻值,并通过查表法将其转换为温度。线性插值可以提高温度计算的精度。查表数据可以根据具体的 NTC 热敏电阻特性曲线来建立。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1906948.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Android的校园二手交易app-计算机毕业设计源码46291

摘要 在大学校园中&#xff0c;学生们的物品更换频繁&#xff0c;有许多闲置物品堆积。对于这些物品&#xff0c;许多学生希望能够有一个平台来方便地交易。随着移动互联网的普及&#xff0c;移动应用已成为校园生活的重要组成部分。其中&#xff0c;校园二手交易平台能够有效地…

(2024,稀疏 MoE,大量小专家,参数高效专家检索 PEER,product key 检索)混合百万专家

Mixture of A Million Experts 公和众与号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0. 摘要 1. 简介 2. 方法 3. 实验 0. 摘要 标准 Transformer 架构中的前馈&#xff08;feedforward&a…

绝区肆--2024 年AI安全状况

前言 随着人工智能系统变得越来越强大和普及&#xff0c;与之相关的安全问题也越来越多。让我们来看看 2024 年人工智能安全的现状——评估威胁、分析漏洞、审查有前景的防御策略&#xff0c;并推测这一关键领域的未来可能如何。 主要的人工智能安全威胁 人工智能系统和应用程…

C++入门基础_cpp

目录 1.C发展历史 2.C版本更新 3. C参考⽂档 4.C的第⼀个程序 5.命名空间(namespace) 5.1 namespace的价值 5.2 namespace的定义 5.3 命名空间使用 6.C输⼊&输出 7. 缺省参数 8.函数重载 9. 引用 9.1 引用的概念和特性 9.2 const引用 9.3 引用与指针的关系 …

学诚教育在线管理系统-计算机毕业设计源码98076

目 录 摘要 1 绪论 1.1 选题背景与意义 1.2开发现状 1.3论文结构与章节安排 2 开发环境及相关技术介绍 2.1 MySQL数据库 2.2 Tomcat服务器 2.3 Java语言 2.4 Spring Cloud框架介绍 3 教育在线管理系统系统分析 3.1 可行性分析 3.1.1 技术可行性分析 3.1.2 经济可…

如何确保工业展厅设计既专业又吸引?三原则详解!

工业是民族发展的基石&#xff0c;它为我们带来了无数的便利和进步&#xff0c;而为了让更多人了解这个至关重要的产业&#xff0c;以及其背后的技术和产品&#xff0c;许多工业性质的企业都致力于通过互动投影、虚拟现实、全息投影等多媒体技术&#xff0c;来打造独具特色的工…

AI视频生成,文字、图片、人像生成视频小程序开发

AI视频生成&#xff0c;文字、图片、人像生成视频小程序开发 AI驱动的多媒体内容创新平台&#xff1a;从文本至视频的一站式生成解决方案。 以下概述集成AI技术的原创视频生成小程序的高级功能框架&#xff0c;旨在为用户提供极致的创作体验。 文本视频化引擎&#xff1a;允…

web端已有项目集成含UI腾讯IM

通过 npm 方式下载 TUIKit 组件,将 TUIKit 组件复制到自己工程的 src 目录下: npm i @tencentcloud/chat-uikit-vue mkdir -p ./src/TUIKit && rsync -av --exclude={node_modules,package.json,excluded-list.txt} ./node_modules/@tencentcloud/chat-uikit-vue/ .…

学数据结构学的很慢,毫无头绪怎么办 ?

这个情况比较正常诶&#xff0c;不用有太大的心理压力。 然后程序设计那个没有学过&#xff0c;而数据结构的前置课程之一就是程序设计&#xff0c;比如栈/队列/树&#xff0c;这些数据结构都要基于代码实现的。我估计是因为你之前缺少学习程序设计的经验&#xff0c;所以学起…

Perforce发布白皮书,解读电动汽车初创公司如何加速进入市场并降低软件开发中的风险和成本

电动汽车&#xff08;EV&#xff09;领域的初创企业正迅速崛起&#xff0c;创新速度显著加快。然而&#xff0c;随着消费者对电动汽车需求的激增&#xff0c;老牌汽车制造商正加速进军这一市场&#xff0c;加剧了行业竞争。为在竞争中生存并发展&#xff0c;电动汽车初创企业必…

JS数据类型检测的方式有哪些 (常用)

typeof 其中数组、对象、null都会被判断为object&#xff0c;其他判断都正确typeof返回的类型都是字符串形式 instanceof instanceof &#xff1a;用于检测一个实例是否属于某个类&#xff0c;通过验证当前类的原型 prototype 是否出现在实例的原型链 __proto__ 上。它不能检测…

读书记录《SQL从小白到大牛》01

读书记录《SQL从小白到大牛》01 接地气的书名&#xff0c;内容应当值得一读。 第一篇 SQL基础 01 一些基础概念 SQL是结构化查询语言&#xff08;Structured Query Language&#xff09;&#xff0c;是一套用来输入、更改和查看关系数据库内容的命令。数据库发展经历三个阶…

花朵短视频:四川江兴川丰科技有限公司

花朵短视频&#xff1a;绽放于屏幕间的自然诗篇 在快节奏的现代生活中&#xff0c;我们常常渴望一抹清新与宁静&#xff0c;以慰藉心灵的疲惫。而花朵短视频&#xff0c;就像是大自然精心编织的一首首无声诗篇&#xff0c;四川江兴川丰科技有限公司通过手机屏幕的方寸之间&…

ArcGIS中国工具(ArcGISCTools)等插件使用体验

ArcGIS中国工具&#xff08;ArcGISCTools&#xff09;的主要功能 1. 接合图表生成 这个功能允许用户生成标准分幅图的行政区边框注记&#xff0c;并在打印时自动加入。这对于需要制作标准地图的用户非常实用。 2. 图框工具 图框工具可以帮助用户创建和管理地图的图框&#…

latex改写字体和字号

文章目录 字体使用宏包设置命令声明命令 字号例子设置特定字号 设置行间距用\setlength{\baselineskip}{24pt}设置\renewcommand{\baselinestretch}{2} \selectfont中文行距&#xff08;{ctex}&#xff09; 补充&#xff1a; 字体 使用宏包 \usepackage{ctex}设置命令 只对确…

ExcelVBA运用Excel的【条件格式】(二)

ExcelVBA运用Excel的【条件格式】&#xff08;二&#xff09; 前面知识点回顾 1. 访问 FormatConditions 集合 Range.FormatConditions 2. 添加条件格式 FormatConditions.Add 方法 语法 表达式。添加 (类型、 运算符、 Expression1、 Expression2) 3. 修改或删除条件…

飞睿智能无线高速uwb安全数据传输模块,低功耗、抗干扰超宽带uwb芯片传输速度技术新突破

在信息化的时代&#xff0c;数据传输的速度和安全性无疑是每个企业和个人都极为关注的话题。随着科技的飞速发展&#xff0c;超宽带&#xff08;Ultra-Wideband&#xff0c;简称UWB&#xff09;技术凭借其性能和广泛的应用前景&#xff0c;逐渐成为了数据传输领域的新星。今天&…

一键高效处理,批量缩放PNG图片,按比例轻松调整,高效工作从此开始!

在数字时代&#xff0c;图片已经成为我们生活中不可或缺的一部分。无论是工作汇报、项目展示还是日常分享&#xff0c;图片都扮演着至关重要的角色。然而&#xff0c;当面对大量需要调整尺寸的PNG图片时&#xff0c;你是否曾经感到过困扰和繁琐&#xff1f; 第一步&#xff0c;…

App Store Connect 《数字服务法》合规性

App Store Connect 《数字服务法》合规性 - 简书 最近Apple开发者圈子比较热门的话题可能就是如标题所言。 如果不进行处理&#xff0c;App Store Connect后台已经给出了对应的影响。 如果你的应用属于国内销售&#xff0c;不走海外市场&#xff0c;那么可以直接选择非交易提…