STM32入门笔记(03): ADC(SPL库函数版)(2)

news2025/1/10 17:16:32

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 库实现低通滤波器:

#include "main.h"

ADC_HandleTypeDef hadc1;

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

#define ALPHA 0.1  // 滤波系数,范围在 0 到 1 之间

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

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)
    {
        previous_filtered_value = Read_ADC_LowPass(&hadc1, previous_filtered_value);

        float V_REF = 3.3;
        float V_out = (previous_filtered_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)
{
    // 时钟配置代码
}

详细说明

  1. 定义滤波系数和变量

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

    • HAL_ADC_Start(hadc); 启动 ADC。
    • HAL_ADC_PollForConversion(hadc, HAL_MAX_DELAY); 等待 ADC 转换完成。
    • uint16_t raw_value = HAL_ADC_GetValue(hadc); 获取 ADC 读取值。
    • float current_filtered_value = ALPHA * raw_value + (1 - ALPHA) * previous_filtered_value; 应用低通滤波公式计算当前滤波值。
  3. 在主循环中持续滤波

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

    • float V_out = (previous_filtered_value / 4095.0) * V_REF; 计算分压电路的输出电压。
    • float V_in = V_out * (R1 + R2) / R2; 反向计算原始输入电压。

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


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/1890590.html

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

相关文章

【分布式数据仓库Hive】HivQL的使用

目录 一、Hive的基本操作 1. 使用Hive创建数据库test 2. 检索数据库&#xff08;模糊查看&#xff09;&#xff0c;检索形如’te*’的数据库 3. 查看数据库test详情 4. 删除数据库test 5. 创建一个学生数据库Stus&#xff0c;在其中创建一个内部表Student&#xff0c;表格…

3ds Max渲染曝光过度怎么办?

3dmax效果图云渲染平台——渲染100 以3ds Max 2025、VR 6.2、CR 11.2等最新版本为基础&#xff0c;兼容fp、acescg等常用插件&#xff0c;同时LUT滤镜等参数也得到了同步支持。 注册填邀请码【7788】可领30元礼包和免费渲染券哦~ 遇到3ds Max渲染过程中曝光过度的问题&#xf…

【刷题汇总--简写单词、dd爱框框、除2!】

C日常刷题积累 今日刷题汇总 - day0031、简写单词1.1、题目1.2、思路1.3、程序实现 - 思路11.4、程序实现 - 思路2(优化) 2、dd爱框框2.1、题目2.2、思路2.3、程序实现 - 蛮力法2.4、程序实现 - 同向双指针(滑动窗口) 3、除2!3.1、题目3.2、思路3.3、程序实现 4、题目链接 今日…

ArcTs布局入门04——相对布局 媒体查询

如果你也对鸿蒙开发感兴趣&#xff0c;加入“Harmony自习室”吧 扫描下面的二维码关注公众号。 本文将探讨相对布局与媒体查询&#xff0c;为啥把他们放到一起呢&#xff1f;主要是因为相对布局在响应式的场景下做得不太好&#xff0c;一般情况下和媒体查询&#xff08;不同尺…

Visual Studio 中的键盘快捷方式

1. Visual Studio 中的键盘快捷方式 1.1. 可打印快捷方式备忘单 1.2. Visual Studio 的常用键盘快捷方式 本部分中的所有快捷方式都将全局应用&#xff08;除非另有指定&#xff09;。 “全局”上下文表示该快捷方式适用于 Visual Studio 中的任何工具窗口。 生成&#xff1…

使用EndNote在Word中插入参考文献,并编辑参考文献样式方法

一、背景 在准备中期报告时&#xff0c;学校给的是Word模板&#xff0c;习惯了Latex排版和添加参考文献的便利后&#xff0c;真不想用word写东西。 之前投《机器人》期刊&#xff08;被拒了&#xff09;和准备开题的时候也是用word写的&#xff0c;当时为方便添加参考文献和定…

【Linux】线程封装与互斥(万字)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 文章目录 前言 C多线程的用法 对原生线程进行一次封装 理解pthread线程 Linux线程互斥 进程线程间的互斥相关背景概念 互斥量mutex 操作共享变量会有问题的售票…

20万内最能打的颜值,小鹏MONA M03开启全球首秀

7月3日&#xff0c;小鹏MONA M03开启全球首秀。这款为年轻用户打造的智能纯电掀背轿跑&#xff0c;以其独特的AI量化美学设计吸引了行业关注。 据「TMT星球」了解&#xff0c;小鹏汽车董事长CEO何小鹏携手造型中心副总裁胡安马洛佩兹&#xff08;JuanMa Lopez&#xff09;共同…

【高中数学/基本不等式】已知:x,y皆大于1,且x+2y=4 求:1/(x-1)+1/(y-1)的最小值为?

【问题来源】 https://www.ixigua.com/7025123539728466469?logTag1c2fd2e305d60e6277ab 之第一题 【问题】 已知&#xff1a;x,y皆大于1&#xff0c;且x2y4 求&#xff1a;1/(x-1)1/(y-1)的最小值为&#xff1f; 【解答】 解&#xff1a; 若将(x2y)/41代入目标式&…

OpenGL3.3_C++_Windows(27)

法线/凹凸贴图 如何让纹理产生更细节的效果&#xff0c;产生凹凸视觉感&#xff1f;解决思路之一&#xff1a;镜面贴图(黑—白&#xff09;&#xff08;&#xff08;diffuse贴图&#xff08;rgba&#xff09;&#xff09;&#xff0c;阻止部分表面被照的更亮&#xff0c;但这并…

鸿蒙数据防泄漏(DLP)【Data Loss Prevention Kit简介】

Data Loss Prevention Kit简介 Data Loss Prevention Kit&#xff08;数据防泄漏服务&#xff0c;简称为DLP&#xff09;&#xff0c;是系统提供的系统级的数据防泄漏解决方案&#xff0c;提供文件权限管理、加密存储、授权访问等能力&#xff0c;数据所有者可以基于帐号认证对…

pjsip环境搭建、编译源码生成.lib库

使用平台&#xff1a; windows qt(5.15.2) vs(2019)x86 pjsip版本以及第三方库使用 pjsip 2.10 ffmpeg4.2.1 sdl2.0.12pjsip源码链接&#xff1a; https://github.com/pjsip/pjproject源码环境配置 首先创建两个文件夹&#xff0c;分别是include、lib其中include放置ff…

亚马逊跟卖卖家还在选品发愁吗!已经有卖家用这种方式选品大卖!

对于亚马逊相信很多卖家都不陌生&#xff0c;也有很多新手卖家涌入&#xff0c;但是进入后就不知道怎么选品了&#xff0c;很多新手卖家是不是天天盯着亚马逊页面的产品&#xff0c;眼花撩乱的&#xff0c;不知道那些产品&#xff0c;能跟卖那些不能跟卖&#xff0c;也有些卖家…

通过卷防水上限,解锁手机的新玩法?IP68之间亦有不同

当手机的日常防水已经成了基本功&#xff0c;防水能力的上限便成了新的赛道。 毕竟再谨慎的人&#xff0c;也可能会有手滑的时候。这个时候&#xff0c;一台有着IP68级防水的手机&#xff0c;就能给你提供一份安心。 【IP68是标准上限&#xff0c;不是手机防水上限】 IP68是…

据阿谱尔统计,全球mRNA原料酶市场预计2024年达到11.98亿美元

Codexis 宣布与 Aldevron 达成协议&#xff0c;授予 Aldevron Codexis 的 Codex HiCap RNA 聚合酶的全球制造和商业化权利。 Applied DNA、Maravai LifeSciences (MRVI) 和 Alphazyme 达成协议&#xff0c;Alphazyme 将扩大 Applied DNA 专有 Linea™ RNA 聚合酶 (RNAP) 的生产…

新声创新20年:无线技术给助听器插上“娱乐”的翅膀

听力损失并非现代人的专利&#xff0c;古代人也会有听力损失。助听器距今发展已经有二百多年了&#xff0c;从当初单纯的声音放大器到如今的全数字时代助听器&#xff0c;助听器发生了翻天覆地的变化&#xff0c;现代助听器除了助听功能&#xff0c;还具有看电视&#xff0c;听…

Character.ai因内容审查流失大量用户、马斯克:Grok-3用了10万块英伟达H100芯片

ChatGPT狂飙160天&#xff0c;世界已经不是之前的样子。 更多资源欢迎关注 1、爆火AI惨遭阉割&#xff0c;1600万美国年轻人失恋&#xff1f;Character.ai被爆资金断裂 美国流行的社交软件Character.ai近期对模型进行大幅度内容审查&#xff0c;导致用户感到失望并开始流失。…

国内外生成式AI教育产品:深度解析教育各环节的智能化教学

随着人工智能技术的飞速发展&#xff0c;教育行业迎来了一场革命性的变革。 2024年&#xff0c;AI教育产品不仅在课堂上大放异彩&#xff0c;更在课前备课、课后辅导、教研支持等各个环节发挥着重要作用。 本文将为您全面总结AI教育产品如何渗透教育的每一个环节&#xff0c;…

PD虚拟机对电脑配置要求高吗 PD虚拟机配置怎么改 Parallels Desktop使用教程

说到虚拟机&#xff0c;很多人会觉得电脑需要很高的配置才能安装。其实不然&#xff0c;有些虚拟机软件的技术已经很成熟&#xff0c;不需要借助电脑的高配置支撑其功能&#xff0c;比如PD软件&#xff0c;很多旧的Mac机型都能安装使用。本文会给大家详细介绍PD虚拟机对电脑配置…

鸿蒙数据防泄漏(DLP)【Data Loss Prevention Kit开发指导】

Data Loss Prevention Kit开发指导 DLP是系统提供的系统级的数据防泄漏解决方案&#xff0c;提供一种称为DLP的文件格式。后缀格式为“原始文件名&#xff08;包含原始文件后缀&#xff09;.dlp”&#xff0c;例如: “test.docx.dlp”&#xff0c;文件由授权凭证和原始文件密文…