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

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

“fc-async”提供了基本的异步处理能力

在开发中,异步处理已经成为提升系统性能和用户体验的常用方式。然而,传统的@Async注解和基础的异步处理工具在面对复杂的任务场景时,存在局限性。这些局限性包括但不限于高并发环境下的稳定性、任务失败后的恢复机制、以及任务的监控和管理。 开源项目“fc-async”提供了基…

【linux】如何扩展磁盘容量(VMware虚拟机)-转载

如何扩展磁盘容量(VMware虚拟机) 一、前置准备工作 扩展虚拟机磁盘前&#xff0c;需要先把虚拟机关机才能进行扩展磁盘操作 1.选择虚拟机设置&#xff0c;如下图所示 2.输入你想扩展的磁盘容量&#xff0c;以本次实操为例&#xff0c;我这里输入的30G&#xff08;具体按照实…

记录配置ubuntu18.04下运行ORBSLAM3的ros接口的过程及执行单目imu模式遇到的问题(详细说明防止忘记)

今天的工作需要自己录制的数据集来验证昨天的标定结果 用ORBSLAM3单目imu模式运行&#xff0c;mentor给的是一个rosbag格式的数据包&#xff0c;配置过程出了几个问题记录一下&#xff0c;沿配置流程写。 一.orbslam3编译安装 1.首先是安装各种依赖 这里不再赘述&#xff0…

STM32设计井下瓦斯检测联网WIFI加Zigbee多路节点协调器传输

目录 目录 前言 一、本设计主要实现哪些很“开门”功能&#xff1f; 二、电路设计原理图 1.电路图采用Altium Designer进行设计&#xff1a; 2.实物展示图片 三、程序源代码设计 四、获取资料内容 前言 本系统基于STM32微控制器和Zigbee无线通信技术&#xff0c;设计了…

华为HCIP——MSTP/RSTP与STP的兼容性

一、MSTP/RSTP与STP的兼容性的原理&#xff1a; 1.BPDU版本号识别&#xff1a;运行MSTP/RSTP协议的交换机会根据收到的BPDU&#xff08;Bridge Protocol Data Unit&#xff0c;桥协议数据单元&#xff09;版本号信息自动判断与之相连的交换机的运行模式。如果收到的是STP BPDU…

Python绘制雪花

文章目录 系列目录写在前面技术需求完整代码代码分析1. 代码初始化部分分析2. 雪花绘制核心逻辑分析3. 窗口保持部分分析4. 美学与几何特点总结 写在后面 系列目录 序号直达链接爱心系列1Python制作一个无法拒绝的表白界面2Python满屏飘字表白代码3Python无限弹窗满屏表白代码4…

第六节、Docker 方式部署指南 github 上项目 mkdocs-material

一、简介 MkDocs 可以同时编译多个 markdown 文件,形成书籍一样的文件。有多种主题供你选择,很适合项目使用。 MkDocs 是快速,简单和华丽的静态网站生成器,可以构建项目文档。文档源文件在 Markdown 编写,使用单个 YAML 配置文件配置。 MkDocs—markdown项目文档工具,…

Spring Boot教程之Spring Boot简介

Spring Boot 简介 接下来一段时间&#xff0c;我会持续发布并完成Spring Boot教程 Spring 被广泛用于创建可扩展的应用程序。对于 Web 应用程序&#xff0c;Spring 提供了 Spring MVC&#xff0c;它是 Spring 的一个广泛使用的模块&#xff0c;用于创建可扩展的 Web 应用程序。…

无线迷踪:陈欣的网络之旅

第一章 陈欣是一名资深的网络工程师&#xff0c;工作在一家领先的科技公司。她的生活平静而有序&#xff0c;直到有一天&#xff0c;公司的无线网络突然出现了严重的问题。员工们的设备频繁断开连接&#xff0c;无法正常使用。这个问题不仅影响了工作效率&#xff0c;还引起了…

ssm129办公用品管理系统开发与设计+jsp(论文+源码)_kaic

毕 业 设 计&#xff08;论 文&#xff09; 题目&#xff1a;办公用品管理系统设计与实现 摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本办公用品管理系统…

OMV7 树莓派 tf卡安装

​ 升级7之后&#xff0c;问题多多&#xff0c;不是docker不行了&#xff0c;就是代理不好使 今天又重装了一遍&#xff0c;用官方的链接&#xff0c;重新再折腾一遍…… 使用raspberry pi imager安装最新版lite OS。 注意是无桌面 Lite版 配置好树莓派初始化设置&#xff0…

【数据结构与算法】查找

文章目录 一.查找二.线性结构的查找2.1顺序查找2.2折半查找2.3分块查找 三.树型结构的查找3.1二叉排序树1.定义2.二叉排序树的常见操作3.性能分析 3.2平衡二叉树1.定义2.平衡二叉树的常见操作3.性能分析 3.3B树1.定义2.B树的相关操作 3.4B树1.定义2.B树与B树的比较 四.散列表1.…

SpringCloud篇(服务保护 - Sentinel)

目录 一、雪崩问题及解决方案 1. 雪崩问题 2. 解决方案 方案一&#xff1a;超时处理 方案二&#xff1a;仓壁模式 方案三&#xff1a;断路器模式 方案四&#xff1a;限流 3. 总结 二、服务保护技术对比 三、Sentinel介绍与安装 1. 初识Sentinel 2. Sentinel 优势 3…

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.贪吃蛇游戏设…

笔记|M芯片MAC (arm64) docker上使用 export / import / commit 构建amd64镜像

很简单的起因&#xff0c;我的东西最终需要跑在amd64上&#xff0c;但是因为mac的架构师arm64&#xff0c;所以直接构建好的代码是没办法跨平台运行的。直接在arm64上pull下来的docker镜像也都是arm64架构。 检查镜像架构&#xff1a; docker inspect 8135f475e221 | grep Arc…

热点更新场景,OceanBase如何实现性能优化

案例背景 这个案例来自一个保险行业的客户&#xff1a;他们的核心系统底层采用了OceanBase数据库作为存储解决方案&#xff0c;然而&#xff0c;在系统上线运行后&#xff0c;出现了一个异常情况&#xff0c;执行简单的主键更新语句时SQL执行时间出现了显著的波动。为了迅速定…

MYSQL_深入理解自连接_图书借阅情况(2/2)

光说不练假把式。这就开门见山——引出我们的自连接实例&#xff1a;图书借阅情况。 题目&#xff1a; 这是一道笔试题目&#xff1a;如果限时5min内完成&#xff0c;同学们可以测试一下自己对于SQL语句的熟练程度。 题目分析&#xff1a; 可以看见这个数据库有三个实体&…

uniapp luch-request 使用教程+响应对象创建

1. 介绍 luch-request 是一个基于 Promise 开发的 uni-app 跨平台、项目级别的请求库。它具有更小的体积、易用的 API 和方便简单的自定义能力。luch-request 支持请求和响应拦截、全局挂载、多个全局配置实例、自定义验证器、文件上传/下载、任务操作、自定义参数以及多拦截器…

MySQL技巧之跨服务器数据查询:基础篇-A数据库与B数据库查询合并--封装到存储过程中

MySQL技巧之跨服务器数据查询&#xff1a;基础篇-A数据库与B数据库查询合并–封装到存储过程中 我们的最终目的是什么&#xff1f;当然的自动执行这些合并操作&#xff01; 上一篇 MySQL技巧之跨服务器数据查询&#xff1a;基础篇-A数据库与B数据库查询合并 我们已经知道怎么合…

解决 IDEA 修改代码重启不生效的问题

前言 在使用 IntelliJ IDEA 进行 Java 项目开发时&#xff0c;有时会遇到一个令人头疼的问题&#xff1a;修改了代码后&#xff0c;重启服务却发现更改没有生效。通常情况下&#xff0c;解决这个问题需要通过 Maven 的 clean 和 compile 命令来强制重新编译&#xff0c;但这显…