STM32f103学习笔记 (十六)- 库函数 AD单通道显示电位器电压 和 AD多通道显示电位器,反射式红外传感器,光敏电阻和热敏电阻电压

目录

一,ADC理论知识

1.ADC简介

2.ADC框图

3.转换模式

4.数据对齐

5.可编程采样时间

6.DMA请求

7. ADC中断标志位

8.校准

二,keil工程 配置AD单通道显示电位器电压

1.接线图

2.复制OLED工程

3.具体步骤

4.代码部分

三,keil工程配置AD多通道显示电压

1.接线图

2.代码部分


一,ADC理论知识

1.ADC简介

1.ADC(Analog-Digital Converter)模拟-数字转换器

2.ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁

3.12位逐次逼近型ADC,1us转换时间

4.输入电压范围:0~3.3V,转换结果范围:0~4095 18个输入通道,可测量16个外部和2个内部信号源

5.规则组和注入组两个转换单元

6.模拟看门狗自动监测输入电压范围 STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道

2.ADC框图

3.转换模式

由单次转换,连续转换和扫描模式,非扫描模式组合

4.数据对齐

分为数据右对齐和数居左对齐

5.可编程采样时间

6.DMA请求

观察上面的ADC框图可知规则通道转换的值储存在一个仅有的数据寄存器中,所以当转换多个规则通道时需要使用 DMA,这可以避免丢失已经存储在ADC_DR寄存器中的数据。 只有在规则通道的转换结束时才产生DMA请求,并将转换的数据从ADC_DR寄存器传输到用户 指定的目的地址。

注: 只有 ADC1 ADC3 拥有 DMA 功能。由 ADC2 转化的数据可以通过双 ADC 模式,利用 ADC1 DMA 功能传输。

7. ADC中断标志位

8.校准

二,keil工程 配置AD单通道显示电位器电压

1.接线图

2.复制OLED工程

之前有发这个工程的网盘

STM32学习笔记(十五)- 库函数 定时器定时中断 OLED显示中断

3.具体步骤

1.开启RCC时钟,包括ADC和GPIO时钟 ADCCLOCK的分频器也要配置

//配置ADCCLK分频器,将APB2的主时钟进行分频输入到ADCCLK
void RCC_ADCCLKConfig(uint32_t RCC_PCLK2);

2.配置GPIO,把需要的用的GPIO配置成模拟输入模式

3.配置输入通道的数据选择器,把左边的输入通道接入到右边的规则组

//ADC规则组通道配置
void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime);

4.配置ADC转换器


//是否允许外部触发转换
void ADC_ExternalTrigConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);

//获取AD转换的数据寄存器,读取转换结果
uint16_t ADC_GetConversionValue(ADC_TypeDef* ADCx);

5.调用ADC_Cmd函数,开启ADC

void ADC_Cmd(ADC_TypeDef* ADCx, FunctionalState NewState);

6.选择触发方式

//选择软件触发
void ADC_SoftwareStartConvCmd(ADC_TypeDef* ADCx, FunctionalState NewState);

7.对ADC进行校准,减少误差

//复位校准
void ADC_ResetCalibration(ADC_TypeDef* ADCx);

//获取复位校准状态
FlagStatus ADC_GetResetCalibrationStatus(ADC_TypeDef* ADCx);

//开始校准
void ADC_StartCalibration(ADC_TypeDef* ADCx);

//获取开始校准状态
FlagStatus ADC_GetCalibrationStatus(ADC_TypeDef* ADCx);

4.代码部分

新建AD.c

#include "stm32f10x.h"                  // Device header

//对AD初始化
void AD_Init(void)
{
	//开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	//ADC分频
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);	//这样ADCCLK=72MHz/6
	
	//配置GPIO
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;	//模拟输入
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	//选择规则组通道
	ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); //最后一个参数是选择采样时间,随你
	
	//用结构体初始化ADC
	ADC_InitTypeDef ADC_InitStructure;
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
	ADC_InitStructure.ADC_NbrOfChannel = 1;
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;
	ADC_Init(ADC1,&ADC_InitStructure);
	
	//开启ADC电源
	ADC_Cmd(ADC1, ENABLE);
	
	//校准,当复位校准状态为SET的时候完成复位校准
	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1) == SET);
	
}

uint16_t AD_GetValue(void)
{
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);
	while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);	//大概会等(55.5+12.5)/(72M/6)= 5.6us
	return ADC_GetConversionValue(ADC1);
}

2.新建AD.h

#ifndef _AD_H
#define _AD_H

#include "stdint.h"
void AD_Init(void);
uint16_t AD_GetValue(void);

#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

uint16_t ADValue;
float Voltage;

int main(void)
{
	OLED_Init();
	AD_Init();
	
	OLED_ShowString(1, 1, "ADValue:");
	OLED_ShowString(2, 1, "Voltage:0.00V");
	
	while(1)
	{
		ADValue = AD_GetValue();
		Voltage = (float)ADValue / 4095 * 3.3;
		
		OLED_ShowNum(1, 9, ADValue, 4);
		OLED_ShowNum(2, 9, Voltage, 1);		//显示整数部分
		OLED_ShowNum(2, 11, (uint16_t)(Voltage * 100) % 100, 2);
		
		Delay_ms(100);
		
	}
}

三,keil工程配置AD多通道显示电压

1.接线图

电位器接PA1,光敏传感器接PA0,反射式红外传感器接PA2,热敏传感器接PA3,OLED显示屏的正负极记得要用跳线连接面包板

本次程序采用单次转换,非扫描模式;在每次触发转换前,手动更改一下列表第一位置的通道(先指定通道,再进行转换)

 

2.代码部分

1.AD.c

#include "stm32f10x.h"                  // Device header

//对AD初始化
void AD_Init(void)
{
	//开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
	
	//ADC分频
	RCC_ADCCLKConfig(RCC_PCLK2_Div6);	//这样ADCCLK=72MHz/6
	
	//配置GPIO
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;	//模拟输入
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	
	
	//用结构体初始化ADC
	ADC_InitTypeDef ADC_InitStructure;
	ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
	ADC_InitStructure.ADC_NbrOfChannel = 1;
	ADC_InitStructure.ADC_ScanConvMode = DISABLE;
	ADC_Init(ADC1,&ADC_InitStructure);
	
	//开启ADC电源
	ADC_Cmd(ADC1, ENABLE);
	
	//校准,当复位校准状态为SET的时候完成复位校准
	ADC_ResetCalibration(ADC1);
	while (ADC_GetResetCalibrationStatus(ADC1) == SET);
	ADC_StartCalibration(ADC1);
	while (ADC_GetCalibrationStatus(ADC1) == SET);
	
}

uint16_t AD_GetValue(uint8_t ADC_Channel)
{
	//选择规则组通道
	ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5); 
	
	ADC_SoftwareStartConvCmd(ADC1, ENABLE);
	while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);	//大概会等(55.5+12.5)/(72M/6)= 5.6us
	return ADC_GetConversionValue(ADC1);
}

2.AD.h

#ifndef _AD_H
#define _AD_H

#include "stdint.h"
void AD_Init(void);
uint16_t AD_GetValue(uint8_t ADC_Channel);

#endif

3.main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "AD.h"

uint16_t AD0, AD1, AD2, AD3;

int main(void)
{
	OLED_Init();
	AD_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);
		AD3 = AD_GetValue(ADC_Channel_3);
		
		OLED_ShowNum(1, 5, AD0, 4);
		OLED_ShowNum(2, 5, AD1, 4);
		OLED_ShowNum(3, 5, AD2, 4);
		OLED_ShowNum(4, 5, AD3, 4);
		
		Delay_ms(100);
		
	}
}

  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值