STM32学习记录0011——AD转换

本文介绍了STM32的ADC模数转换器的基本概念和使用方法,包括ADC控制器数量、通道配置、转换模式、时钟设置和寄存器操作。通过一个实际应用案例展示了如何初始化ADC,读取多个通道的转换值,并进行平均值计算显示。代码示例中涵盖了从配置ADC到显示ADC转换结果的完整流程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、STM32 ADC

1.1参考资料

《STM32不完全手册》第20章
《STM32中文参考手册》第11章
芯片数据手册ADC部分+GPIO部分

1.2 什么是ADC

ADC是Analog-to- Converter 的缩写,指模/数转换器,将连续的模拟信号转换为离散的数字信号的器件,一般使用电压值来表示数字化。

1.3 怎么学

实话说,看完中文参考手册ADC部分,就是一个感觉,听君一席话,就是听君一席话…
学习的时候管制一下这几个问题

  • 几个ADC控制器
  • 几位什么型的AD
  • 那些引脚有ADC功能
  • 参考电压问题
  • 注入通道/规则通道是什么,有什么区别
  • 有哪些转换模式(单次、连续、扫描、间断)
  • 时钟
  • 寄存器

1、基本信息

在这里插入图片描述

2、引脚对应表

在这里插入图片描述

3、电压问题

在这里插入图片描述

4、框图

在这里插入图片描述
注释:
1、规则通道正常进入,按照代码顺序执行的AD采集,最多16个通道,注入通道通过中断事件进入的AD采集,最多4个通道

5、关于模式

1、单次转换在这里插入图片描述
2、连续转换
在这里插入图片描述
3、扫描模式
在这里插入图片描述
4、间断模式
在这里插入图片描述
设置不同的序列后将他们组成一个小组,然后进行小组的AD采集

二、寄存器

1、ADC_CR1

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

2、ADC_CR2

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

3、ADC_SMPR1和ADC_SMPR2

在这里插入图片描述

在这里插入图片描述
可以独立设置每一个通道的采用时间
ADC转换时间
在这里插入图片描述

4、ADC_SQR1、ADC_SQR2、ADC_SQR3和ADC_JSQR

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

5、ADC_JDRx和ADC_DR

在这里插入图片描述

6、ADC_SR

在这里插入图片描述

三、库函数配置

1、void ADC_Init(ADC_TypeDef* ADCx, ADC_InitTypeDef* ADC_InitStruct);

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
不同参数的有效输入可以查看固件库函数说明
在这里插入图片描述

2、void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);

在这里插入图片描述

3、void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState)

使用软件转换函数
在这里插入图片描述

4、ADC_RegularChannelConfig

规则通道配置函数
在这里插入图片描述
在这里插入图片描述

5、uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx)

获取ADC转换结果的值
在这里插入图片描述

四、应用

目标:实现4个AD采集并且显示

1、初始化

在这里插入图片描述

在这里插入图片描述

2、读取转换值

在这里插入图片描述

3、多次采集优化结果

在这里插入图片描述

4、主函数

在这里插入图片描述
在这里插入图片描述

5、代码

(1)主函数

#include "led.h"
#include "delay.h"
#include "sys.h"
#include "usart.h"
#include "lcd.h"
#include "adc.h"

   	
 int main(void)
 { 
	u16 adcx;
	float temp;
	u16 adcx2;
	float temp2;
	u16 adcx3;
	float temp3;
	u16 adcx4;
	float temp4;
	delay_init();	    	 //延时函数初始化	  
	uart_init(9600);	 	//串口初始化为9600
	LED_Init();		  		//初始化与LED连接的硬件接口
 	LCD_Init();
 	Adc_Init();		  		//ADC初始化	    
	POINT_COLOR=RED;//设置字体为红色 
	LCD_ShowString(60,50,200,16,16,"Mini STM32");	
	LCD_ShowString(60,70,200,16,16,"ADC TEST");	
	LCD_ShowString(60,90,200,16,16,"lixingye");
	LCD_ShowString(60,110,200,16,16,"2022/04/28");	
	//显示提示信息
	POINT_COLOR=BLUE;//设置字体为蓝色
	LCD_ShowString(60,130,200,16,16,"ADC_CH1_VAL:");	      
	LCD_ShowString(60,150,200,16,16,"ADC_CH1_VOL:0.000V");	  
	LCD_ShowString(60,170,200,16,16,"ADC_CH2_VAL:");	      
	LCD_ShowString(60,190,200,16,16,"ADC_CH2_VOL:0.000V");	 
	LCD_ShowString(60,210,200,16,16,"ADC_CH3_VAL:");	      
	LCD_ShowString(60,230,200,16,16,"ADC_CH3_VOL:0.000V");	 
	LCD_ShowString(60,250,200,16,16,"ADC_CH4_VAL:");	      
	LCD_ShowString(60,270,200,16,16,"ADC_CH4_VOL:0.000V");	   
	while(1)
	{
		adcx=Get_Adc_Average(ADC_Channel_1,10);
		LCD_ShowxNum(156,130,adcx,4,16,0);//显示ADC的值
		temp=(float)adcx*(3.3/4096);
		adcx=temp;
		LCD_ShowxNum(156,150,adcx,1,16,0);//显示电压值
		temp-=adcx;
		temp*=1000;
		LCD_ShowxNum(172,150,temp,3,16,0X80);
		LED0=!LED0;
		delay_ms(250);	
		
		adcx2=Get_Adc_Average(ADC_Channel_2,10);
		LCD_ShowxNum(156,170,adcx2,4,16,0);//显示ADC的值
		temp2=(float)adcx2*(3.3/4096);
		adcx2=temp2;
		LCD_ShowxNum(156,190,adcx2,1,16,0);//显示电压值
		temp2-=adcx2;
		temp2*=1000;
		LCD_ShowxNum(172,190,temp2,3,16,0X80);
		LED0=!LED0;
		delay_ms(250);	
		
		adcx3=Get_Adc_Average(ADC_Channel_3,10);
		LCD_ShowxNum(156,210,adcx3,4,16,0);//显示ADC的值
		temp3=(float)adcx3*(3.3/4096);
		adcx3=temp3;
		LCD_ShowxNum(156,230,adcx3,1,16,0);//显示电压值
		temp3-=adcx3;
		temp3*=1000;
		LCD_ShowxNum(172,230,temp3,3,16,0X80);
		LED0=!LED0;
		delay_ms(250);	
		
		adcx4=Get_Adc_Average(ADC_Channel_4,10);
		LCD_ShowxNum(156,250,adcx4,4,16,0);//显示ADC的值
		temp4=(float)adcx4*(3.3/4096);
		adcx4=temp4;
		LCD_ShowxNum(156,270,adcx4,1,16,0);//显示电压值
		temp4-=adcx4;
		temp4*=1000;
		LCD_ShowxNum(172,270,temp4,3,16,0X80);
		LED0=!LED0;
		delay_ms(250);	
		

	}											    
}	

(2)adc.c

 #include "adc.h"
 #include "delay.h"

	   
		   
//初始化ADC
//这里我们仅以规则通道为例
//我们默认将开启通道0~3																	   
void  Adc_Init(void)
{ 	
	ADC_InitTypeDef ADC_InitStructure; 
	GPIO_InitTypeDef GPIO_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1	, ENABLE );	  //使能ADC1通道时钟
 

	RCC_ADCCLKConfig(RCC_PCLK2_Div6);   //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M

	//PA1 作为模拟通道输入引脚                         
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;		//模拟输入引脚
	GPIO_Init(GPIOA, &GPIO_InitStructure);	
		//PA2 作为模拟通道输入引脚                         
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;		//模拟输入引脚
	GPIO_Init(GPIOA, &GPIO_InitStructure);	
		//PA3 作为模拟通道输入引脚                         
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;		//模拟输入引脚
	GPIO_Init(GPIOA, &GPIO_InitStructure);	
		//PA4 作为模拟通道输入引脚                         
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;		//模拟输入引脚
	GPIO_Init(GPIOA, &GPIO_InitStructure);	

	ADC_DeInit(ADC1);  //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值

	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;	//ADC工作模式:ADC1和ADC2工作在独立模式
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;	//模数转换工作在单通道模式
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;	//模数转换工作在单次转换模式
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;	//转换由软件而不是外部触发启动
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;	//ADC数据右对齐
	ADC_InitStructure.ADC_NbrOfChannel = 1;	//顺序进行规则转换的ADC通道的数目
	ADC_Init(ADC1, &ADC_InitStructure);	//根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器   
	
		ADC_InitStructure.ADC_NbrOfChannel = 2;	//顺序进行规则转换的ADC通道的数目
	ADC_Init(ADC1, &ADC_InitStructure);	//根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器  
		ADC_InitStructure.ADC_NbrOfChannel = 3;	//顺序进行规则转换的ADC通道的数目
	ADC_Init(ADC1, &ADC_InitStructure);	//根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器  
		ADC_InitStructure.ADC_NbrOfChannel = 4;	//顺序进行规则转换的ADC通道的数目
	ADC_Init(ADC1, &ADC_InitStructure);	//根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器  
	

  
	ADC_Cmd(ADC1, ENABLE);	//使能指定的ADC1
	
	ADC_ResetCalibration(ADC1);	//使能复位校准  
	 
	while(ADC_GetResetCalibrationStatus(ADC1));	//等待复位校准结束
	
	ADC_StartCalibration(ADC1);	 //开启AD校准
 
	while(ADC_GetCalibrationStatus(ADC1));	 //等待校准结束
 
//	ADC_SoftwareStartConvCmd(ADC1, ENABLE);		//使能指定的ADC1的软件转换启动功能

}				  
//获得ADC值
//ch:通道值 0~3
u16 Get_Adc(u8 ch)   
{
  	//设置指定ADC的规则组通道,一个序列,采样时间
	ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5 );	//ADC1,ADC通道,采样时间为239.5周期	  			    
  
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);		//使能指定的ADC1的软件转换启动功能	
	 
	while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束

	return ADC_GetConversionValue(ADC1);	//返回最近一次ADC1规则组的转换结果
}

u16 Get_Adc_Average(u8 ch,u8 times)
{
	u32 temp_val=0;
	u8 t;
	for(t=0;t<times;t++)
	{
		temp_val+=Get_Adc(ch);
		delay_ms(5);
	}
	return temp_val/times;
} 	 

(3)adc.h

#ifndef __ADC_H
#define __ADC_H	
#include "sys.h"


void Adc_Init(void);
u16  Get_Adc(u8 ch); 
u16 Get_Adc_Average(u8 ch,u8 times); 
 
#endif 

五、总结

在这里插入图片描述

这里使用的一个是最简单的模式了,还是要继续去研究其他模式怎么样使用的!

### STM32 ADC 学习教程和笔记 #### 一、STM32 ADC 基本概念 STM32F103ZET6 单片机配备有三个12位精度的模拟数字转换器(ADC),总共提供18个通道,能够处理来自外部的16个信号源以及两个内部信号源。这些通道支持多种工作模式,包括但不限于单次转换、连续转换、扫描模式和间断模式。值得注意的是,为了确保正常运作,ADC模块所需的输入时钟频率不应超过14MHz,该时钟是从APB2总线上的PCLK2经过预分频得到的[^3]。 #### 二、ADC 软件启动机制分析 对于通过软件指令来发起一次新的转换操作而言,在某些情况下可能会涉及到查询`ADC_GetSoftwareStartConvStatus()`函数的状态返回值;然而实际上此方法并不常用,因为其功能较为有限,并未被广泛应用于实际项目开发当中[^2]。 #### 三、触发方式的选择 除了上述提到的纯软件控制之外,还可以利用硬件事件作为触发条件来进行自动化的数据采集过程。这类触发源通常来源于定时器产生的中断请求或者其他外设发出的相关脉冲信号。此外,整个AD转换流程依赖于由RCC配置好的专用时钟源——即所谓的ADC_CLK,它是驱动逐次逼近型A/D变换的核心动力源泉[^4]。 #### 四、性能参数考量 假设当前使用的ADC时钟频率设定为最大允许值14MHz,则完成一轮完整的采样加量化所需时间大约等于1微秒左右。具体计算公式如下所示: \[T_{conv}=(SMPR+12.5)\times\frac{1}{f_{adc}} \] 其中\( SMPR \)代表样本保持阶段所占用的时间长度,默认设置下取值为1.5个周期数[^5]。 ```python # Python代码片段用于验证理论计算结果 def calculate_conversion_time(f_adc, smpr=1.5): """Calculate the conversion time based on given parameters.""" t_conv = (smpr + 12.5) * (1 / f_adc) return round(t_conv, ndigits=9) print(calculate_conversion_time(14e6)) # Output should be close to 1 us ```
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

li星野

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值