当前位置: 首页 > news >正文

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

相关文章:

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模数…...

解决IntelliJ IDEA的Plugins无法访问Marketplace去下载插件

勾选Auto-detect proxy setting并填入 https://plugins.jetbrains.com 代理URL&#xff0c;可以先做检查连接&#xff1a;...

react 如何修改弹出的modal的标题

原来标题的样子&#xff1a; 修改为&#xff1a; 实现方式&#xff1a; <Modal title<span>股价趋势/{this.state.pccode}</span> visible{this.state.isPriceModalOpen} style{{ top: 20 }} width{1320} height{400} footer{null} onCancel{()>this.hideMo…...

C#中的二维数组的应用:探索物理含义与数据结构的奇妙融合

在C#编程中&#xff0c;二维数组&#xff08;或矩阵&#xff09;是一种重要的数据结构&#xff0c;它不仅能够高效地存储和组织数据&#xff0c;还能通过其行、列和交叉点&#xff08;备注&#xff1a;此处相交处通常称为“元素”或“单元格”&#xff0c;代表二维数组中的一个…...

HTML5拖拽API学习 托拽排序和可托拽课程表

文章目录 前言拖拽API核心概念拖拽式使用流程例子注意事项综合例子&#x1f330; 可拖拽课程表拖拽排序 前言 前端拖拽功能让网页元素可以通过鼠标或触摸操作移动。HTML5 提供了标准的拖拽API&#xff0c;简化了拖放操作的实现。以下是拖拽API的基本使用指南&#xff1a; 拖拽…...

内容补充页(相关公式解释)

from 学习日记_20241117_聚类方法&#xff08;高斯混合模型&#xff09; 学习日记_20241117_聚类方法&#xff08;高斯混合模型&#xff09; 公式 P ( Z k ) π k P(Zk) \pi_k P(Zk)πk​ 在高斯混合模型 (GMM) 中&#xff0c;公式 P ( Z k ) π k P(Zk) \pi_k P(Zk…...

vue中动态渲染静态图片资源

不报错且f12查看元素的时候&#xff0c;显示的src说明已经渲染到html的src上&#xff0c;但是就是不显示在页面上 原因 在vue上&#xff0c;动态渲染静态图片资源&#xff08;比如从assets文件夹加载的图片&#xff09;需要注意打包工具对静态资源的解析方式 由于vue2的脚手…...

管伊佳ERP,原名华夏ERP,一个简约易上手的国产ERP系统

JSH_ERP&#xff08;管伊佳ERP&#xff09;是一款开源、模块化的企业资源计划系统&#xff0c;旨在为中小企业提供高效的管理工具。它基于SpringBoot框架和SaaS模式&#xff0c;支持进销存、财务、生产等业务模块&#xff0c;包括零售、采购、销售、仓库和报表管理。 核心特点…...

学习虚幻C++开发日志——委托(持续更新中)

委托 官方文档&#xff1a;Delegates and Lamba Functions in Unreal Engine | 虚幻引擎 5.5 文档 | Epic Developer Community | Epic Developer Community 简单地说&#xff0c;委托就像是一个“函数指针”&#xff0c;但它更加安全和灵活。它允许程序在运行时动态地调用不…...

开窗函数 - first_value/last_value

1、开窗函数是什么&#xff1f; 开窗函数用于为行定义一个窗口&#xff08;这里的窗口是指运算将要操作的行的集合&#xff09;&#xff0c;它对一组值进行操作&#xff0c;不需要使用 GROUP BY 子句对数据进行分组&#xff0c;能够在同一行中同时返回基础行的列和聚合列。 2、…...

「一」HarmonyOS端云一体化概要

关于作者 白晓明 宁夏图尔科技有限公司董事长兼CEO、坚果派联合创始人 华为HDE、润和软件HiHope社区专家、鸿蒙KOL、仓颉KOL 华为开发者学堂/51CTO学堂/CSDN学堂认证讲师 开放原子开源基金会2023开源贡献之星 「目录」 「一」HarmonyOS端云一体化概要 「二」体验HarmonyOS端云一…...

nodejs21: 快速构建自定义设计样式Tailwind CSS

Tailwind CSS 是一个功能强大的低级 CSS 框架&#xff0c;只需书写 HTML 代码&#xff0c;无需书写 CSS&#xff0c;即可快速构建美观的网站。 1. 安装 Tailwind CSS React 项目中安装 Tailwind CSS&#xff1a; 1.1 安装 Tailwind CSS 和相关依赖 安装 Tailwind CSS: npm…...

从JSON数据提取嵌套字段并转换为独立列的简洁方法

从JSON数据提取嵌套字段并转换为独立列的简洁方法 在数据处理和数据分析的日常工作中&#xff0c;我们经常遇到复杂的嵌套数据结构&#xff0c;特别是嵌入在JSON字段中的数据。这些数据往往需要解析并展开成独立的列&#xff0c;以便后续分析和建模。本文将详细介绍如何在Pyth…...

湘潭大学软件工程算法设计与分析考试复习笔记(四)

回顾 湘潭大学软件工程算法设计与分析考试复习笔记&#xff08;一&#xff09;湘潭大学软件工程算法设计与分析考试复习笔记&#xff08;二&#xff09;湘潭大学软件工程算法设计与分析考试复习笔记&#xff08;三&#xff09; 前言 现在是晚上十一点&#xff0c;我平时是十…...

特征交叉-DeepCross Network学习

一 tensorflow官方实现 tensorflow的官方实现已经是V2版本 class Cross(tf.keras.layers.Layer):"""Cross Layer in Deep & Cross Network to learn explicit feature interactions.Args:projection_dim: int&#xff0c;低秩矩阵的维度&#xff0c;应该小…...

stm32cubemx+VSCODE+GCC+makefile 开发环境搭建

title: stm32cubemxVSCODEGCCmakefile 开发环境搭建 tags: FreertosHalstm32cubeMx 文章目录 内容往期内容导航第一步准备环境vscode 插件插件配置点灯 内容 往期内容导航 第一步准备环境 STM32CubeMXVSCODEMinGWOpenOcdarm-none-eabi-gcc 然后把上面下载的软件 3 4 5 bin 文…...

Go语言中的Defer机制详解与示例

在Go语言中&#xff0c;defer是一个关键字&#xff0c;用于确保资源的清理和释放&#xff0c;特别是在函数中创建的资源。defer语句会将其后的函数调用推迟到包含它的函数即将返回时执行。这使得defer成为处理文件关闭、数据库连接释放、解锁等资源清理操作的理想选择。 Defer…...

H.265流媒体播放器EasyPlayer.js H5流媒体播放器如何验证视频播放是否走硬解

随着技术的不断进步和5G网络的推广&#xff0c;中国流媒体播放器行业市场规模以及未来发展趋势都将持续保持稳定的增长&#xff0c;并将在未来几年迎来新的发展机遇。流媒体播放器将继续作为连接内容创作者和观众的重要桥梁&#xff0c;推动数字媒体产业的创新和发展。 EasyPla…...

ms-hot目录

1. ms-hot1...

vulfocus在线靶场:骑士cms_cve_2020_35339:latest 速通手册

目录 一、启动环境&#xff0c;访问页面&#xff0c;ip:端口号/index.php?madmin,进入后台管理页面&#xff0c;账号密码都是adminadmin 二、进入之后&#xff0c;根据图片所示&#xff0c;地址后追加一下代码&#xff0c;保存修改 ​三、新开标签页访问&#xff1a;①ip:端…...

AI Large Language Model

AI 的 Large Language model LLM , 大语言模型&#xff1a; 是AI的模型&#xff0c;专门设计用来处理自然语言相关任务。它们通过深度学习和庞大的训练数据集&#xff0c;在理解和生成自然语言文本方面表现出色。常见的 LLM 包括 OpenAI 的 GPT 系列、Google 的 PaLM 和 Meta…...

React Native的`react-native-reanimated`库中的`useAnimatedStyle`钩子来创建一个动画样式

React Native的react-native-reanimated库中的useAnimatedStyle钩子来创建一个动画样式&#xff0c;用于一个滑动视图的每个项目&#xff08;SliderItem&#xff09;。useAnimatedStyle钩子允许你根据动画值&#xff08;在这个例子中是scrollX&#xff09;来动态地设置组件的样…...

FastJson反序列化漏洞(CVE-2017-18349)

漏洞原理 原理就不多说了&#xff0c;可以去看我这篇文章&#xff0c;已经写得很详细了。 Java安全—log4j日志&FastJson序列化&JNDI注入-CSDN博客 影响版本 FastJson<1.2.24 复现过程 这里我是用vulfocus.cn这个漏洞平台去复现的&#xff0c;比较方便&#x…...

【优选算法篇】分治乾坤,万物归一:在重组中窥见无声的秩序

文章目录 分治专题&#xff08;二&#xff09;&#xff1a;归并排序的核心思想与进阶应用前言、第二章&#xff1a;归并排序的应用与延展2.1 归并排序&#xff08;medium&#xff09;解法&#xff08;归并排序&#xff09;C 代码实现易错点提示时间复杂度和空间复杂度 2.2 数组…...

C++:探索AVL树旋转的奥秘

文章目录 前言 AVL树为什么要旋转&#xff1f;一、插入一个值的大概过程1. 插入一个值的大致过程2. 平衡因子更新原则3. 旋转处理的目的 二、左单旋1. 左单旋旋转方式总处理图2. 左单旋具体会遇到的情况3. 左单旋代码总结 三、右单旋1. 右单旋旋转方式总处理图2. 右单旋具体会遇…...

2. Django中的URL调度器 (自定义路径转换器)

在 Django 中&#xff0c;URL 路由通常使用路径转换器&#xff08;path converters&#xff09;来匹配和捕获 URL 中的特定模式&#xff0c;例如整数、字符串或 slug 等。默认情况下&#xff0c;Django 提供了一些内置的路径转换器&#xff0c;如 <int>、<str>、&l…...

深度学习:神经网络中线性层的使用

深度学习&#xff1a;神经网络中线性层的使用 在神经网络中&#xff0c;线性层&#xff08;也称为全连接层或密集层&#xff09;是基础组件之一&#xff0c;用于执行输入数据的线性变换。通过这种变换&#xff0c;线性层可以重新组合输入数据的特征&#xff0c;并将其映射到新…...

【刷题】算法设计题+程序设计题【2】2019-2024

11.202019年真题*2BST二叉排序树分裂、双向冒泡排序 2019 真题 【2019 1】编写算法&#xff0c;将一棵二叉排序树 分解成两棵二叉排序树 t1和t2&#xff0c;使得t1中的所有结点关键字的值都小于x&#xff0c;t2中所有结点关键字都大于x。 typedef struct BSTNode{int data;str…...

搭建es环境

centos7搭建elasticsearch环境 首先考虑使用 Docker 来安装 Elasticsearch、Kibana 和 Logstash。在安装过程中&#xff0c;可能会遇到一些问题&#xff0c;但通过适当的方法可以解决。 docker pull docker.elastic.co/elasticsearch/elasticsearch:8.14.3 首先创建一个网络&a…...

阿里云和七牛云对象存储区别和实现

七牛云对象存储操作&#xff08;QiniuUtil&#xff09; 配置&#xff1a;使用 com.qiniu.storage.Configuration 类来配置上传设置&#xff0c;如指定区域&#xff08;Region&#xff09;和分片上传版本。上传管理器&#xff1a;通过 UploadManager 类来处理文件上传。认证&am…...

怎么来自己做网站优化/百度手机app

DRM实例教程DRM是一个显示驱动框架&#xff0c;也就是把功能封装成 open/close/ioctl 等标准接口&#xff0c;应用程序调用这些接口来驱动设备&#xff0c;显示数据。我们这里将从使用的角度来看看&#xff0c;怎么验证和使用DRM驱动。DRM设备节点DRM驱动会在/dev/dri下创建3个…...

网站建设用什么书/如何进行搜索引擎营销

4月17日AMD公布了第一季度财报。报告显示&#xff0c;AMD第一季度净亏损为1.8亿美元&#xff0c;去年同期的净亏损为2000万美元。该股盘后股价暴跌8%。作为今天业绩发布的一部分&#xff0c;AMD的CFO Devinder Kumar表示&#xff0c;“AMD公司退出密集服务器业务&#xff0c;即…...

上海免费网站建设/宁波seo教学

作者&#xff1a;张志朋出处&#xff1a;https://blog.52itstyle.com题记工作也有几多年了&#xff0c;无论是身边遇到的还是耳间闻到的&#xff0c;多多少少也积攒了自己的一些经验和思考&#xff0c;当然&#xff0c;博主并没有太多接触高大上的分布式架构实践&#xff0c;相…...

资金盘做网站/什么是网站推广?

上文《嵌入式Qt开发-移植到ARM开发板 》介绍了Qt程序的移植&#xff0c;本文再说下如何使开发板Qt程序使用触摸屏或USB方式进行交互。 之前刚把一个qt程序移植到arm板上成功运行显示时就开心的要死&#xff08;Linux用了很久&#xff0c;但从未搞过Qt界面开发&#xff09;&…...

网站推广的图片/百度热门搜索排行榜

环境Centos 7.4Python 2.7Pip 2.7 MySQL-python 1.2.5 Elasticsearc 6.3.1Elasitcsearch6.3.2知识点调用Python Elasticsearh APIPython Mysqldb使用DSL查询与聚合Python 列表操作代码#!/usr/bin/env python# -*- coding: utf-8 -*-#minyt 2018.9.1#获取24小时内出现的模块次数…...

wordpress 清除/郑州网站营销推广

定义&#xff1a;定义一系列算法&#xff0c;将它们一个个封装起来&#xff0c;并且使他们之间可以相互替换。本模式使得算法可以独立于使用它的客户而变化。 类型&#xff1a;对象行为型模式 类图&#xff1a; 策略模式是对算法的封装&#xff0c;把一系列的算法分别封装到对应…...