stm32下的ADC转换(江科协 HAL版)

十二. ADC采样

文章目录

    • 十二. ADC采样
      • 12.1 ADC的采样原理
      • 12.2 STM32的采样基本过程
        • 1.引脚与GPIO端口的对应关系
        • 2.ADC规则组的四种转换模式(**)
          • 2.2 关于转换模式与配置之间的关系
      • 12.3 ADC的时钟
      • 12.4 代码实现(ADC单通道 & ADC多通道)
        • 1. 单通道采样
        • 2. 多通道采样

19.ADC模数转换器知识点+AD单通道&AD多通道应用程序示例_ad模数转换器-CSDN博客

​ 常见的IO口采样通常只能表示高(1)低(0)电平而ADC(Aalog-to-Digital Converter)采样的功能即类似于电压表的功能,也就是将模拟值转化为数据值的数值(即可对范围内人机模拟值进行采量)。即建立模拟电路到数字电路的桥梁, 当然像PWM则是实现数字到模拟的桥梁,而真正的DAC则只是在一些特定的领域应用。

  • 模拟量:指实际一系列的物理值,可在连续范围内变化
  • 数值量:一种离散值

​ ADC的进行检测时靠输入通道进行的,stm32下有18个输入通道,其中16个外部,2个内部。且可与模拟看门狗进行后续的联动检测。这里的STM32F103C8T6 的ADC资源:ADC1、ADC2,10个外部输入通道。

12.1 ADC的采样原理

ADC的核心原理是怎样的,是怎样实现采样的过程的?

常见的ADC采样结构

​ 这种算是常见的逐次ADC采样的元器件示意图,其采样通道有(IN0 ~ IN7),采样后,将采样的电压与ADC内部的DAC模块产生的电压进行比较(在比较器中进行),再根据比较结果不断调整DAC的值,直到不断逼近最终的采样值,而这DAC调整的过程就由SAR寄存器来完成的(通常调整方法为二分查找法),最后再将结果通过(D0 ~ D7进行输出)。一般情况下,ADC的电压范围与ADC的供电范围是一致的

Stm32下的采样结构图

​ 通常来讲的话,手册里的每个外设都有其对应的结构图,而这个结构图也是至关重要的。

​ 整个结构图可分为如上几个区域,其中外部模拟信号输入由2号区域(ADCx_INx)输入,然后途径AD转换器(由注入通道和规则通道组成),其不同之处在于注入组一次能选中16个通道而规则组一次只能整4个通道(“各自上菜”)。各自组转换完成后再将完成结果传输到各自的数据寄存器(6号区域),其中规则组一次只能上一个值,注入组有4个数据寄存器一次能上4个结果。在一般使用时,主要是使用规则组,规则组寄存器通常通过DMA请求和DMA进行联动(需要注意的是只有ADC1和ADC3能开启DMA请求),及时将值保存到内存中。此外模拟看门狗还通常和AD的数据寄存器相连,若超过其阈值电压,狗子则会“乱叫”并申请一个中断之后通向NVIC。这里主要使用的是规则组。

需要注意的是这里的ADC时钟ADCCLK来自RCC的APB2时钟,其最大为14MHz,ADC的预分频只能选择6分频(72/6 = 12MHz)和8分频两种值。

ADC的基本结构

​ 编程主要根据基本结构来进行代码的撰写

12.2 STM32的采样基本过程

1.引脚与GPIO端口的对应关系

​ 这里的ADC1与ADC2的引脚相同,采用的双ADC模式,也就是交叉模式,交叉模式就像你打拳一样,左手打一圈、右手打一圈快速交叉地打拳,那打击的频率肯定比一个拳头打得快。

2.ADC规则组的四种转换模式(**)

ADC转换模式有哪些,各自定义是什么?

代码配置中的关系是什么?

扫描模式需要注意的点是什么?

​ ADC的转换模式是指从模拟信号到数字信号转换的方法,这里的转换主要是针对规则组和注入组的相关通道而言,主要有以下几种:

  • 单次转换 非扫描模式
  • 单次转换 扫描模式
  • 连续转换 非扫描模式
  • 连续转换 扫描模式

单次转换: 执行一次转换就停下来,直到下次转换的触发

连续转换: 一次转换结束后马上开始新的转换

*扫描模式:*ADC的所有的通道(规则和注入的)均选中,在每个通道上执行单次转换

间断模式(非扫描模式): 算是对连续模式的一种补充,在选择转换通道组中由触发信号开启新的不同组的转换

这里的扫描和非扫描是专门针对不止单一通道而言的。而连续或非连续则是针对采样的频率。需要注意的是这里的通道是针对“规则”通道而言,也就是说每一组物理通道可组成一个规则通道(最多可一个规则通道下有16个物理通道),而扫描模式针对的是多个规则通道,并不是多个物理采集通道。

2.2 关于转换模式与配置之间的关系

ADC单通道

​ 若只进行单次采样,则选择单次转换非扫描,若进行多次采样则连续转换,非扫描。

ADC多通道

​ 若需多个通道进行单次采样,则选择单次转换,扫描模式,若需多次则连续转换扫描模式,一句话就是多通道必须使能扫描模式

采样后的数据分为右对齐和左对齐两种方式:

12.3 ADC的时钟

ADC的整个转换步骤为: 采样 (保持) 量化 编码 其中量化和编码即ADC逐次比较的过程,采样和保持也可看做前采样过程,这个周期算是自己设置。而量化和编码通常为12.5个周期。也就是说最后ADC时钟为:

12.4 代码实现(ADC单通道 & ADC多通道)

​ 采样后的范围为(0~4095),需再将其转换为自己所需要的值。

0.配置相关

1. 单通道采样

1.1.相关初始化

ADC_HandleTypeDef hadc1;

/* ADC1 init function */
void MX_ADC1_Init(void)
{

  /* USER CODE BEGIN ADC1_Init 0 */

  /* USER CODE END ADC1_Init 0 */

  ADC_ChannelConfTypeDef sConfig = {0};

  /* USER CODE BEGIN ADC1_Init 1 */

  /* USER CODE END ADC1_Init 1 */
  /** Common config
  */
  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;
  //hadc1.Init.ExternalTrigConv
  
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }
  /** Configure Regular Channel
  */
  sConfig.Channel = ADC_CHANNEL_0;
  sConfig.Rank = ADC_REGULAR_RANK_1;
  //sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  //采样周期的设置
  sConfig.SamplingTime  = ADC_SAMPLETIME_55CYCLES_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN ADC1_Init 2 */
	/* 5. 校准ADC,若不进行校准则采样值存在错误,大概率会比规定范围(0~4095)更小 */
   HAL_ADCEx_Calibration_Start(&hadc1);  // 启动校准
  /* USER CODE END ADC1_Init 2 */

}

void HAL_ADC_MspInit(ADC_HandleTypeDef* adcHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(adcHandle->Instance==ADC1)
  {
  /* USER CODE BEGIN ADC1_MspInit 0 */

  /* USER CODE END ADC1_MspInit 0 */
    /* ADC1 clock enable */
    __HAL_RCC_ADC1_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    GPIO_InitStruct.Pin = GPIO_PIN_0;
    //GPIO口模式选择模数模式
    GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  /* USER CODE BEGIN ADC1_MspInit 1 */

  /* USER CODE END ADC1_MspInit 1 */
  }
}

1.2 采样函数

uint16_t  AD_GetValue()
{
  uint16_t adcvalue = 0;

  //启动一次转换
  HAL_ADC_Start(&hadc1);

  //等待转换完成,10ms为超时时间
  if(HAL_ADC_PollForConversion(&hadc1, 10) == HAL_OK)
  {
    adcvalue  = HAL_ADC_GetValue(&hadc1);
  }

  HAL_ADC_Stop(&hadc1);
  return adcvalue;
}

1.3 主函数

int main(void)
{
  
  HAL_Init();

  
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  //MX_USART1_UART_Init();
	OLED_Init();
  
	OLED_ShowString(1, 1,"Success!");
  OLED_ShowString(2, 1, "ADValue:");
	OLED_ShowString(3, 1, "Voltage:0.00V");
  while (1)
  {
    ADValue = AD_GetValue();
    /* USER CODE END WHILE */
		Voltage = ((float)ADValue / 4095) * 3.3f;
    OLED_ShowNum(2, 9, ADValue, 4);

    //显示转换后数据的整数部分与小数部分
    OLED_ShowNum(3, 9, Voltage, 1);
		OLED_ShowNum(3, 11, (uint16_t)(Voltage*100)%100, 2);
    HAL_Delay(100);

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

​ 这里需要尤其注意的是,采样在进行初始化操作时,一定需要进行HAL_ADCEx_Calibration_Start(&hadc1); // 启动校准,否则可能出现采样的最大值小于范围4095。

2. 多通道采样

物理层面的多通道与配置层的多通道的区别是什么?

需要注意的是这里的多通道指的是物理层面的多通道采集,而前面配置时的多通道则是指配置规则组的多通道。

1.单一规则通道多物理通道转换

​ 需要注意的是这里的采样通道数 hadc1.Init.NbrOfConversion = 1;是指多规则组通道,需设置为1,扫描模式也不用打开。

#include "adc.h"


ADC_HandleTypeDef hadc1;

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

 
  //ADC_ChannelConfTypeDef sConfig = {0};

  hadc1.Instance = ADC1;
  //是否开启扫描模式
  hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
  //连续转换是否开启
  hadc1.Init.ContinuousConvMode = DISABLE;
  hadc1.Init.DiscontinuousConvMode = ENABLE;
  //模数转换时使用软件触发,不需要外部触发
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  //数据采取右对齐方式
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  //采样的通道数
  hadc1.Init.NbrOfConversion = 1;
  //hadc1.Init.ExternalTrigConv
  
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }
  /** 
   * 这里是配置每个“规则”通道,配置三个通道进行多通道采样
  */
  //sConfig.Channel = ADC_CHANNEL_0;
  //表示规则组,每个规则组可配置16个物理通道
  sConfig.Rank = ADC_REGULAR_RANK_1;
  //sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  //采样周期的设置
  sConfig.SamplingTime  = ADC_SAMPLETIME_55CYCLES_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN ADC1_Init 2 */
	/* 5. 校准ADC,若不进行校准则采样值存在错误,大概率会比规定范围(0~4095)更小 */
   
  //启用校准
  HAL_ADCEx_Calibration_Start(&hadc1);
}

void HAL_ADC_MspInit(ADC_HandleTypeDef* adcHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(adcHandle->Instance==ADC1)
  {
  /* USER CODE BEGIN ADC1_MspInit 0 */

  /* USER CODE END ADC1_MspInit 0 */
    /* ADC1 clock enable */
    __HAL_RCC_ADC1_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    GPIO_InitStruct.Pin = GPIO_PIN_0 | GPIO_PIN_1 | GPIO_PIN_2;
    //GPIO口模式选择模数模式
    GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  /* USER CODE BEGIN ADC1_MspInit 1 */

  /* USER CODE END ADC1_MspInit 1 */
  }
}

void HAL_ADC_MspDeInit(ADC_HandleTypeDef* adcHandle)
{

  if(adcHandle->Instance==ADC1)
  {
  /* USER CODE BEGIN ADC1_MspDeInit 0 */

  /* USER CODE END ADC1_MspDeInit 0 */
    /* Peripheral clock disable */
    __HAL_RCC_ADC1_CLK_DISABLE();

    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    HAL_GPIO_DeInit(GPIOA, GPIO_PIN_0);

  /* USER CODE BEGIN ADC1_MspDeInit 1 */

  /* USER CODE END ADC1_MspDeInit 1 */
  }
}

/* USER CODE BEGIN 1 */
uint16_t  AD_GetValue(uint8_t ADC_Channel)
{
  uint16_t adcvalue = 0;
  //ADC_ChannelConfTypeDef sConfig = {0};
  //sConfig.Rank = ADC_REGULAR_RANK_1;

  sConfig.Channel = ADC_Channel;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);

  //HAL_ADCEx_Calibration_Start(&hadc1);
  //启动一次转换
  HAL_ADC_Start(&hadc1);

  //等待转换完成,10ms为超时时间
  if(HAL_ADC_PollForConversion(&hadc1, 10) == HAL_OK)
  {
    adcvalue  = HAL_ADC_GetValue(&hadc1);
  }

  HAL_ADC_Stop(&hadc1);
  return adcvalue;
}

/* USER CODE END 1 */

main

int main(void)
{
  
  HAL_Init();

  
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  //MX_USART1_UART_Init();
	OLED_Init();
  
	OLED_ShowString(1, 1, "AD0:");
	OLED_ShowString(2, 1, "AD1:");
	OLED_ShowString(3, 1, "AD2:");
	//OLED_ShowString(4, 1, "AD3:");
  while (1)
  {
    AD0   =  AD_GetValue(ADC_CHANNEL_0); 
    AD1   =  AD_GetValue(ADC_CHANNEL_1);
    AD2   =  AD_GetValue(ADC_CHANNEL_2);
    //显示转换后数据的整数部分与小数部分
    OLED_ShowNum(1, 5, AD0, 4);
    OLED_ShowNum(2, 5, AD1, 4);
    OLED_ShowNum(3, 5, AD2, 4);
    HAL_Delay(100);

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

2.多规则物理通道的采样

​ 当连续采样开关关闭时,这里的每次采样均是用软件去手动控制,这样才能保证到每个通道均采样到真正的物理值,否则则只会采样到一个通道的值(因为不连续设置本质是指采样一次)。

#include "adc.h"


ADC_HandleTypeDef hadc1;

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

 
  //ADC_ChannelConfTypeDef sConfig = {0};

  hadc1.Instance = ADC1;
  //是否开启扫描模式
  //hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
  hadc1.Init.ScanConvMode = ADC_SCAN_ENABLE;
  //连续转换是否开启
  hadc1.Init.ContinuousConvMode = DISABLE;
  hadc1.Init.DiscontinuousConvMode = ENABLE;
  //模数转换时使用软件触发,不需要外部触发
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  //数据采取右对齐方式
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  //采样的通道数
  hadc1.Init.NbrOfConversion = 3;
  //hadc1.Init.ExternalTrigConv
  
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }
  /** 
   * 这里是配置每个“规则”通道,配置三个通道进行多通道采样
  */
  sConfig.Channel = ADC_CHANNEL_0;
  //表示规则组,每个规则组可配置16个物理通道
  sConfig.Rank = ADC_REGULAR_RANK_1;
  //sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  //采样周期的设置
  sConfig.SamplingTime  = ADC_SAMPLETIME_55CYCLES_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN ADC1_Init 2 */
	/* 5. 校准ADC,若不进行校准则采样值存在错误,大概率会比规定范围(0~4095)更小 */
   
  /* USER CODE END ADC1_Init 2 */
  sConfig.Channel = ADC_CHANNEL_1;
  sConfig.Rank = ADC_REGULAR_RANK_2;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);

  sConfig.Channel = ADC_CHANNEL_2;
  sConfig.Rank = ADC_REGULAR_RANK_3;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);

 // HAL_ADCEx_Calibration_Start(&hadc1);  // 启动校准
  //启用校准
  HAL_ADCEx_Calibration_Start(&hadc1);
}

void HAL_ADC_MspInit(ADC_HandleTypeDef* adcHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(adcHandle->Instance==ADC1)
  {
  /* USER CODE BEGIN ADC1_MspInit 0 */

  /* USER CODE END ADC1_MspInit 0 */
    /* ADC1 clock enable */
    __HAL_RCC_ADC1_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    GPIO_InitStruct.Pin = GPIO_PIN_0 | GPIO_PIN_1 | GPIO_PIN_2;
    //GPIO口模式选择模数模式
    GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  /* USER CODE BEGIN ADC1_MspInit 1 */

  /* USER CODE END ADC1_MspInit 1 */
  }
}

void HAL_ADC_MspDeInit(ADC_HandleTypeDef* adcHandle)
{

  if(adcHandle->Instance==ADC1)
  {
  /* USER CODE BEGIN ADC1_MspDeInit 0 */

  /* USER CODE END ADC1_MspDeInit 0 */
    /* Peripheral clock disable */
    __HAL_RCC_ADC1_CLK_DISABLE();

    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    HAL_GPIO_DeInit(GPIOA, GPIO_PIN_0);

  /* USER CODE BEGIN ADC1_MspDeInit 1 */

  /* USER CODE END ADC1_MspDeInit 1 */
  }
}
void AD_GetValue2(uint16_t *adcvalue)
{
  //HAL_ADC_Start(&hadc1);  // 启动ADC转换

  sConfig.Channel = ADC_CHANNEL_0;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);

  HAL_ADC_Start(&hadc1);  // 启动ADC转换
  // 等待转换完成
  if (HAL_ADC_PollForConversion(&hadc1, 10) == HAL_OK)
  {
    adcvalue[0] = HAL_ADC_GetValue(&hadc1);  // 读取第一个通道
  }

  // 如果需要获取下一个通道的值,可以重新配置通道并启动转换
  sConfig.Channel = ADC_CHANNEL_1;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);
  HAL_ADC_Start(&hadc1);
  if (HAL_ADC_PollForConversion(&hadc1, 10) == HAL_OK)
  {
    adcvalue[1] = HAL_ADC_GetValue(&hadc1);  // 读取第二个通道
  }

  sConfig.Channel = ADC_CHANNEL_2;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);
  HAL_ADC_Start(&hadc1);
  if (HAL_ADC_PollForConversion(&hadc1, 10) == HAL_OK)
  {
    adcvalue[2] = HAL_ADC_GetValue(&hadc1);  // 读取第三个通道
  }

  HAL_ADC_Stop(&hadc1);  // 停止ADC转换
}

/* USER CODE END 1 */

当连续采样开关打开时则可采用以下写法:

#include "adc.h"


ADC_HandleTypeDef hadc1;

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

 
  //ADC_ChannelConfTypeDef sConfig = {0};

  hadc1.Instance = ADC1;
  //是否开启扫描模式
  //hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
  hadc1.Init.ScanConvMode = ADC_SCAN_ENABLE;
  //连续转换是否开启
  hadc1.Init.ContinuousConvMode = ENABLE;
  hadc1.Init.DiscontinuousConvMode = DISABLE;
  //模数转换时使用软件触发,不需要外部触发
  hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
  //数据采取右对齐方式
  hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
  //采样的通道数
  hadc1.Init.NbrOfConversion = 3;
  //hadc1.Init.ExternalTrigConv
  
  if (HAL_ADC_Init(&hadc1) != HAL_OK)
  {
    Error_Handler();
  }
  /** 
   * 这里是配置每个“规则”通道,配置三个通道进行多通道采样
  */
  sConfig.Channel = ADC_CHANNEL_0;
  //表示规则组,每个规则组可配置16个物理通道
  sConfig.Rank = ADC_REGULAR_RANK_1;
  //sConfig.SamplingTime = ADC_SAMPLETIME_1CYCLE_5;
  //采样周期的设置
  sConfig.SamplingTime  = ADC_SAMPLETIME_55CYCLES_5;
  if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
  {
    Error_Handler();
  }
  /* USER CODE BEGIN ADC1_Init 2 */
	/* 5. 校准ADC,若不进行校准则采样值存在错误,大概率会比规定范围(0~4095)更小 */
   
  /* USER CODE END ADC1_Init 2 */
  sConfig.Channel = ADC_CHANNEL_1;
  sConfig.Rank = ADC_REGULAR_RANK_2;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);

  sConfig.Channel = ADC_CHANNEL_2;
  sConfig.Rank = ADC_REGULAR_RANK_3;
  HAL_ADC_ConfigChannel(&hadc1, &sConfig);

 // HAL_ADCEx_Calibration_Start(&hadc1);  // 启动校准
  //启用校准
  HAL_ADCEx_Calibration_Start(&hadc1);
}

void HAL_ADC_MspInit(ADC_HandleTypeDef* adcHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct = {0};
  if(adcHandle->Instance==ADC1)
  {
  /* USER CODE BEGIN ADC1_MspInit 0 */

  /* USER CODE END ADC1_MspInit 0 */
    /* ADC1 clock enable */
    __HAL_RCC_ADC1_CLK_ENABLE();

    __HAL_RCC_GPIOA_CLK_ENABLE();
    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    GPIO_InitStruct.Pin = GPIO_PIN_0 | GPIO_PIN_1 | GPIO_PIN_2;
    //GPIO口模式选择模数模式
    GPIO_InitStruct.Mode = GPIO_MODE_ANALOG;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

  /* USER CODE BEGIN ADC1_MspInit 1 */

  /* USER CODE END ADC1_MspInit 1 */
  }
}

void HAL_ADC_MspDeInit(ADC_HandleTypeDef* adcHandle)
{

  if(adcHandle->Instance==ADC1)
  {

    __HAL_RCC_ADC1_CLK_DISABLE();

    /**ADC1 GPIO Configuration
    PA0-WKUP     ------> ADC1_IN0
    */
    HAL_GPIO_DeInit(GPIOA, GPIO_PIN_0);

  }
}

/* USER CODE BEGIN 1 */
void  AD_GetValue2(uint16_t * adcvalue)
{
  
  //启动一次转换
  HAL_ADC_Start(&hadc1);

  //等待转换完成,10ms为超时时间
  for (int i = 0; i < 3; i++)
  {
    /* code */
      if(HAL_ADC_PollForConversion(&hadc1, 10) == HAL_OK)
      {
        adcvalue[i]  = HAL_ADC_GetValue(&hadc1);
      }
  }
  

  HAL_ADC_Stop(&hadc1);
  //return adcvalue;
}
/* USER CODE END 1 */

main

#include "main.h"
#include "adc.h"
#include "usart.h"
#include "gpio.h"
#include "i2c.h"
#include "OLED.h"


void SystemClock_Config(void);

uint16_t ADValue;			//定义AD值变量
float Voltage;				//定义电压变量
uint16_t AD0, AD1, AD2;
uint16_t ADValue2[3] = {0};
/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  
  HAL_Init();

  
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_ADC1_Init();
  //MX_USART1_UART_Init();
	OLED_Init();
  
	OLED_ShowString(1, 1, "AD0:");
	OLED_ShowString(2, 1, "AD1:");
	OLED_ShowString(3, 1, "AD2:");
	//OLED_ShowString(4, 1, "AD3:");
  while (1)
  {
    //ADValue2 = AD_GetValue2();数组变量是不能直接进行赋值的
		AD_GetValue2(ADValue2);
    //显示转换后数据的整数部分与小数部分
    OLED_ShowNum(1, 5, ADValue2[0], 4);
    OLED_ShowNum(2, 5, ADValue2[1], 4);
    OLED_ShowNum(3, 5, ADValue2[2], 4);
    HAL_Delay(100);

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

/**
  * @brief System Clock Configuration
  * @retval None
  */
void SystemClock_Config(void)
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
  RCC_PeriphCLKInitTypeDef PeriphClkInit = {0};

  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
  /** Initializes the CPU, AHB and APB buses clocks
  */
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;

  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
  PeriphClkInit.PeriphClockSelection = RCC_PERIPHCLK_ADC;
  PeriphClkInit.AdcClockSelection = RCC_ADCPCLK2_DIV6;
  if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInit) != HAL_OK)
  {
    Error_Handler();
  }
}

/* USER CODE BEGIN 4 */

/* USER CODE END 4 */

/**
  * @brief  This function is executed in case of error occurrence.
  * @retval None
  */
void Error_Handler(void)
{
  /* USER CODE BEGIN Error_Handler_Debug */
  /* User can add his own implementation to report the HAL error return state */
  __disable_irq();
  while (1)
  {
  }
  /* USER CODE END Error_Handler_Debug */
}

#ifdef  USE_FULL_ASSERT
/**
  * @brief  Reports the name of the source file and the source line number
  *         where the assert_param error has occurred.
  * @param  file: pointer to the source file name
  * @param  line: assert_param error line source number
  * @retval None
  */
void assert_failed(uint8_t *file, uint32_t line)
{
  /* USER CODE BEGIN 6 */
  /* User can add his own implementation to report the file name and line number,
     ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
  /* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */


总结: 在进行ADC转换过程中,其进行转换的原理是二分查找式方法逐步逼近。在进行相关配置操作时,一共有四大模式,其中进行规则多通道采样时需打开扫描模式,另外需要注意的是这里的多通道指的是多规则通道而非物理多通道,在采样初始化时还需进行校准,否则数据准确性会受影响,最后真正采样数会受到一系列外在微小的干扰。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

124. 二叉树中的最大路径和【 力扣(LeetCode) 】

文章目录 零、原题链接一、题目描述二、测试用例三、解题思路四、参考代码 零、原题链接 124. 二叉树中的最大路径和 一、题目描述 二叉树中的 路径 被定义为一条节点序列&#xff0c;序列中每对相邻节点之间都存在一条边。同一个节点在一条路径序列中 至多出现一次 。该路径…

【安全科普】NUMA防火墙诞生记

一、我为啥姓“NUMA” 随着网络流量和数据包处理需求的指数增长&#xff0c;曾经的我面对“高性能、高吞吐、低延迟”的要求&#xff0c;逐渐变得心有余而力不足。 多CPU技术应运而生&#xff0c;SMP&#xff08;对称多处理&#xff09;和NUMA&#xff08;非一致性内存访问&a…

免费送源码:Java+Springboot+MySQL Springboot多租户博客网站的设计 计算机毕业设计原创定制

Springboot多租户博客网站的设计 摘 要 博客网站是当今网络的热点&#xff0c;博客技术的出现使得每个人可以零成本、零维护地创建自己的网络媒体&#xff0c;Blog站点所形成的网状结构促成了不同于以往社区的Blog文化&#xff0c;Blog技术缔造了“博客”文化。本文课题研究的“…

数字IC后端实现之Innovus specifyCellEdgeSpacing和ICC2 set_placement_spacing_rule的应用

昨天帮助社区IC训练营学员远程协助解决一个Calibre DRC案例。通过这个DRC Violation向大家分享下Innovus和ICC2中如何批量约束cell的spacing rule。 数字IC后端手把手实战教程 | Innovus verify_drc VIA1 DRC Violation解析及脚本自动化修复方案 下图所示为T12nm A55项目的Ca…

IntelliJ+SpringBoot项目实战(七)--在SpringBoot中整合Redis

Redis是项目开发中必不可少的缓存工具。所以在SpringBoot项目中必须整合Redis。下面是Redis整合的步骤&#xff1a; &#xff08;1&#xff09;因为目前使用openjweb-sys作为SpringBoot的启动应用&#xff0c;所以在openjweb-sys模块的application-dev.yml中增加配置参数&…

深挖C++赋值

详解赋值 const int a 10; int b a;&a 0x000000b7c6afef34 {56496} &a 0x000000b7c6afef34 {10} 3. &b 0x000000b7c6afef54 {10} 总结&#xff1a; int a 10 是指在内存中&#xff08;栈&#xff09;中创建一个int &#xff08;4 byte&#xff09;大小的空间…

java八股-jvm入门-程序计数器,堆,元空间,虚拟机栈,本地方法栈,类加载器,双亲委派,类加载执行过程

文章目录 PC Register堆虚拟机栈方法区(Metaspace元空间双亲委派机制类加载器 类装载的执行过程 PC Register 程序计数器&#xff08;Program Counter Register&#xff09;是 Java 虚拟机&#xff08;JVM&#xff09;中的一个组件&#xff0c;它在 JVM 的内存模型中扮演着非常…

11.12机器学习_特征工程

四 特征工程 1 特征工程概念 特征工程:就是对特征进行相关的处理 一般使用pandas来进行数据清洗和数据处理、使用sklearn来进行特征工程 特征工程是将任意数据(如文本或图像)转换为可用于机器学习的数字特征,比如:字典特征提取(特征离散化)、文本特征提取、图像特征提取。 …

STL序列式容器之list

相较于vector的连续性空间&#xff0c;list相对比较复杂&#xff1b;list内部使用了双向环形链表的方式对数据进行存储&#xff1b;list在增加元素时&#xff0c;采用了精准的方式分配一片空间对数据及附加指针等信息进行存储&#xff1b; list节点定义如下 template<clas…

算法日记 26-27day 贪心算法

接下来的题目有些地方比较相似。需要注意多个条件。 题目&#xff1a;分发糖果 135. 分发糖果 - 力扣&#xff08;LeetCode&#xff09; n 个孩子站成一排。给你一个整数数组 ratings 表示每个孩子的评分。 你需要按照以下要求&#xff0c;给这些孩子分发糖果&#xff1a; 每…

Linux下编译MFEM

本文记录在Linux下编译MFEM的过程。 零、环境 操作系统Ubuntu 22.04.4 LTSVS Code1.92.1Git2.34.1GCC11.4.0CMake3.22.1Boost1.74.0oneAPI2024.2.1 一、安装依赖 二、编译代码 附录I: CMakeUserPresets.json {"version": 4,"configurePresets": [{&quo…

Pytest-Bdd-Playwright 系列教程(9):使用 数据表(DataTable 参数) 来传递参数

Pytest-Bdd-Playwright 系列教程&#xff08;9&#xff09;&#xff1a;使用 数据表&#xff08;DataTable 参数&#xff09; 来传递参数 前言一、什么是 datatable 参数&#xff1f;Gherkin 表格示例 二、datatable 参数的基本使用三、完整代码和运行效果完整的测试代码 前言 …

C语言项⽬实践-贪吃蛇

目录 1.项目要点 2.窗口设置 2.1mode命令 2.2title命令 2.3system函数 2.Win32 API 2.1 COORD 2.2 GetStdHandle 2.3 CONSOLE_CURSOR_INFO 2.4 GetConsoleCursorInfo 2.5 SetConsoleCursorInfo 2.5 SetConsoleCursorPosition 2.7 GetAsyncKeyState 3.贪吃蛇游戏设…

使用 Prompt API 与您的对象聊天

tl;dr&#xff1a;GET、PUT、PROMPT。现在&#xff0c;可以使用新的 PromptObject API 仅使用自然语言对存储在 MinIO 上的对象进行总结、交谈和提问。在本文中&#xff0c;我们将探讨这个新 API 的一些用例以及代码示例。 赋予动机&#xff1a; 对象存储和 S3 API 的无处不在…

Oracle OCP认证考试考点详解082系列19

题记&#xff1a; 本系列主要讲解Oracle OCP认证考试考点&#xff08;题目&#xff09;&#xff0c;适用于19C/21C,跟着学OCP考试必过。 91. 第91题&#xff1a; 题目 解析及答案&#xff1a; 关于 Oracle 数据库中的索引及其管理&#xff0c;以下哪三个陈述是正确的&#x…

脑机接口、嵌入式 AI 、工业级 MR、空间视频和下一代 XR 浏览器丨RTE2024 空间计算和新硬件专场回顾

这一轮硬件创新由 AI 引爆&#xff0c;或许最大受益者仍是 AI&#xff0c;因为只有硬件才能为 AI 直接获取最真实世界的数据。 在人工智能与硬件融合的新时代&#xff0c;实时互动技术正迎来前所未有的创新浪潮。从嵌入式系统到混合现实&#xff0c;从空间视频到脑机接口&…

Element UI如何实现按需导入--Vue3篇

前言 在使用 Element UI 时&#xff0c;按需导入&#xff08;即按需引入&#xff09;是一个常见的需求&#xff0c;尤其是在构建大型应用时。按需导入可以显著减少打包后的文件体积&#xff0c;提升应用的加载速度。本文将详细介绍如何在项目中实现 Element UI 的按需导入&…

【设计模式】行为型模式(五):解释器模式、访问者模式、依赖注入

《设计模式之行为型模式》系列&#xff0c;共包含以下文章&#xff1a; 行为型模式&#xff08;一&#xff09;&#xff1a;模板方法模式、观察者模式行为型模式&#xff08;二&#xff09;&#xff1a;策略模式、命令模式行为型模式&#xff08;三&#xff09;&#xff1a;责…

.NET 9.0 中 System.Text.Json 的全面使用指南

以下是一些 System.Text.Json 在 .NET 9.0 中的使用方式&#xff0c;包括序列化、反序列化、配置选项等&#xff0c;并附上输出结果。 基本序列化和反序列化 using System; using System.Text.Json; public class Program {public class Person{public string Name { get; se…

《InsCode AI IDE:编程新时代的引领者》

《InsCode AI IDE&#xff1a;编程新时代的引领者》 一、InsCode AI IDE 的诞生与亮相二、独特功能与优势&#xff08;一&#xff09;智能编程体验&#xff08;二&#xff09;多语言支持与功能迭代 三、实际应用与案例&#xff08;一&#xff09;游戏开发案例&#xff08;二&am…