STM32_7(ADC)

news2024/12/23 9:28:20

一、ADC

  • ADC(Analog-Digital Converter)模拟-数字转换器
  • ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量,建立模拟电路到数字电路的桥梁
  • 12位逐次逼近型ADC,1us转换时间
  • 输入电压范围:0~3.3V,转换结果范围:0~4095
  • 18个输入通道,可测量16个外部和2个内部信号源
  • 规则组和注入组两个转换单元
  • 模拟看门狗自动监测输入电压范围(有个需求,如果光线或温度高于或低于某个阈值,就会执行一些操作)
  • STM32F103C8T6 ADC资源:ADC1、ADC2,10个外部输入通道

1. 逐次逼近型ADC

        IN0~IN7,通过通道选择开关,选中一路,输入到比较器进行转换和比较。首先是一个电压比较器,它可以判断两个输入信号电压的大小关系,输出一个高低电平指示谁大谁大小。 这个比较器有两个输入端,一个是待测电压,另一个是DAC电压输出端,DAC是数模转换器,给它一个数据,他就能输出数据对应的电压。一个是外部输入端的未知编码电压,一个是DAC输出的已知编码电压,他们同时输入电压比较器,进行大小判断。如果DAC输出的电压比较大,就调小DAC数据,如果DAC输出电压小,就增大DAC数据,直到DAC输出的电压和外部通道输入的电压近似相等,那么DAC输入的数据就是外部电压的编码数据。

2. STM32ADC框图

        使用规则通道配合DMA就可以不用担心数据被覆盖。

        EOC是规则组的完成信号,JEOC是注入组的完成信号,这两个信号都会再状态寄存器里置一个标志位,读取标志位就可以知道是否转换结束。

3. ADC基本结构图

4. 输入通道

5. 转换模式

① 单次转换,非扫描模式

        比如序列1为通道2,之后就可以触发转换,ADC对通道2进行模数转换,过一定的时间后,转换完成,转换结果放在数据寄存器里,同时给EOC标志位置1。

② 连续转换,非扫描模式

        连续转换与单次转换不同的是,它再一次转换结束后不会停止,而是立刻开始下一轮的转换,之后一直持续下去。

③ 单次转换,扫描模式

        单次转换跟上面的单次转换非扫描模式大致一样,但是扫描模式就会用到菜单列表,可以连续对选中的位置进行转换,转换结果都放在数据寄存器里,但为了防止数据被覆盖,需要用到DMA及时把数据挪走,7个通道转换完成之后,产生EOC信号,转换结束。

④ 连续转换,扫描模式

6. 触发控制 

7. 数据对齐

① 数据右对齐

② 数据左对齐

        12位ADC,但数据寄存器有16位,就分为数据右对齐和左对齐。一般情况下是选择数据右对齐,数据左对齐会让误差变大,因为一位就是要乘2,进4位就是乘16,所以一般不用,除非就用简单的判断,把数据的高8位取出来,舍弃后4位精度。

8. 转换时间

  • AD转换的步骤:采样,保持,量化,编码
  • STM32 ADC的总转换时间为:TCONV = 采样时间 + 12.5个ADC周期
  • 例如:当ADCCLK=14MHz,采样时间为1.5个ADC周期,TCONV = 1.5 + 12.5 = 14个ADC周期 = 1μs

9. 校准

  • ADC有一个内置自校准模式。校准可大幅减小因内部电容器组的变化而造成的准精度误差。校准期间,在每个电容器上都会计算出一个误差修正码(数字值),这个码用于消除在随后的转换中每个电容器上产生的误差
  • 建议在每次上电后执行一次校准
  • 启动校准前, ADC必须处于关电状态超过至少两个ADC时钟周期

10. 硬件电路

二、代码部分

1. 单通道配置代码

#include "Bsp_ADC.h"

void Bsp_ADC_Init(void)
{

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);                            // 1.时钟配置
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

    RCC_ADCCLKConfig(RCC_PCLK2_Div6);                                               // 2.ADC分频,ADC最大14M,接近来的时候是72M,所以要分频到14M以下

    GPIO_InitTypeDef GPIO_InitStructure;                                            // 3.GPIO配置
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);     // 4.配置规则组
    // ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 2, ADC_SampleTime_55Cycles5); // 相当于填充菜单列表方法

    ADC_InitTypeDef ADC_InitStructure;                                              // 5.配置ADC
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;    
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                          // 数据对齐
    ADC_InitStructure.ADC_NbrOfChannel = 1;                                         
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;             // 外部触发转换选择
    //ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;                           // 连续转换模式
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; 
    ADC_Init(ADC1, &ADC_InitStructure);

    ADC_Cmd(ADC1, ENABLE);                                                          // 6.开启ADC
    
    ADC_ResetCalibration(ADC1);                                                     // 7.校准ADC
    while (ADC_GetResetCalibrationStatus(ADC1) == SET);
    ADC_StartCalibration(ADC1);
    while (ADC_GetCalibrationStatus(ADC1) == SET);

    ADC_SoftwareStartConvCmd(ADC1, ENABLE);                                         // ADC软件启动转换,因为开启了连续转换模式,所以只开启一次即可

    // 如果需要看门狗和中断,则需要额外配置(开启看门狗和中断配置)
}

uint16_t ADC_GetValue(void)
{
    //ADC_SoftwareStartConvCmd(ADC1, ENABLE);                                       // ADC软件启动转换
    //while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);                        // 等待ADC读取EOC标志位为1;   并且也不需要判断标志位了
    return ADC_GetConversionValue(ADC1);                                            // 返回ADC1的值
}

2. 多通道配置代码

#include "Bsp_ADC.h"

void Bsp_ADC_Init(void)
{

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE);                            // 1.时钟配置
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

    RCC_ADCCLKConfig(RCC_PCLK2_Div6);                                               // 2.ADC分频,ADC最大14M,接近来的时候是72M,所以要分频到14M以下

    GPIO_InitTypeDef GPIO_InitStructure;                                            // 3.GPIO配置
    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_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5);  // 4.配置规则组
    // ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 2, ADC_SampleTime_55Cycles5); // 相当于填充菜单列表方法

    ADC_InitTypeDef ADC_InitStructure;                                              // 5.配置ADC
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;    
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;                          // 数据对齐
    ADC_InitStructure.ADC_NbrOfChannel = 1;                                         
    ADC_InitStructure.ADC_ScanConvMode = DISABLE;
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;             // 外部触发转换选择
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;                             // 连续转换模式
    // ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;                   
    ADC_Init(ADC1, &ADC_InitStructure);

    ADC_Cmd(ADC1, ENABLE);                                                          // 6.开启ADC
    
    ADC_ResetCalibration(ADC1);                                                     // 7.校准ADC
    while (ADC_GetResetCalibrationStatus(ADC1) == SET);
    ADC_StartCalibration(ADC1);
    while (ADC_GetCalibrationStatus(ADC1) == SET);

    // ADC_SoftwareStartConvCmd(ADC1, ENABLE);                                      // ADC软件启动转换,因为开启了连续转换模式,所以只开启一次即可

    // 如果需要看门狗和中断,则需要额外配置(开启看门狗和中断配置)
}

/* 读取ADC的每个通道,在主程序里读取4次即可完成多通道 */
uint16_t ADC_GetValue(uint8_t ADC_Channel)
{
    ADC_SoftwareStartConvCmd(ADC1, ENABLE);                                         // ADC软件启动转换
    while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET);                          // 等待ADC读取EOC标志位为1;   并且也不需要判断标志位了
    ADC_RegularChannelConfig(ADC1, ADC_Channel, 1, ADC_SampleTime_55Cycles5);
    return ADC_GetConversionValue(ADC1);                                            // 返回ADC1的值
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

命名空间、字符串、布尔类型、nullptr、类型推导

面向过程语言:C ——> 重视求解过程 面向对象语言:C ——> 重视求解的方法 面向对象的三大特征:封装、继承和多态 C 和 C 在语法上的区别 1、命名空间(用于解决命名冲突问题) 2、函数重载和运算符重载&#xf…

C语言进阶之路-基本数据小怪篇

目录 一、学习目标: 二、数据基本类型 整型 浮点型 / 实型 字符 字符串 布尔型数据 三、重要的杂七杂八知识点 常量与变量 标准输入 sizeof运算符: 类型转换 数据类型的本质 整型数据尺寸 可移植性整型 拿下第一个C语言程序 总结 一、学…

MySQL数据库主从集群搭建

快捷查看指令 ctrlf 进行搜索会直接定位到需要的知识点和命令讲解(如有不正确的地方欢迎各位小伙伴在评论区提意见,博主会及时修改) MySQL数据库主从集群搭建 主从复制,是用来建立一个和主数据库完全一样的数据库环境&#xff0c…

win10安装pytorch(py39)

cuda≤11.6,观察控制面板 观察torch对应cuda版本 https://download.pytorch.org/whl/torch/ 安装cuda11.6.0 CUDA Toolkit Archive | NVIDIA Developer cmd输入nvcc -V 编辑国内镜像源 .condarc anaconda prompt输入 查看环境 conda env list 安装py3.9…

LedControl 库说明文档

LedControl 库最初是为基于 8 位 AVR 处理器的 Arduino 板编写的。但由于该代码不使用处理器的任何复杂的内部功能,因此具有高度可移植性,并且应该在任何支持 和 功能的 Arduino(类似)板上pinMode()运行digitalWrite() 。 单个 M…

中电金信:守【政】创新,探路保险数字化转型“新范式”

11月23日,CIIP2023中国保险科技创新合作大会在京举办。大会汇集保险科技领域行业专家、学者、国内外头部险企及保险科技公司负责人等各界人士,立足保险行业高质量发展和创新驱动理念,寻找行业数字化转型新动能、新视角,为保险科技…

【基础知识】AB软件RSLinx如何实现OPC通讯组态

哈喽,大家好,我是雷工。 在上一节了解了什么是RSLinx?以及RSLinx Lite、RSLinx Classice、RSLinx Professional、RSLinx Gateway几个版本的特点。 本节了解AB的RSLinx如何实现OPC组态。 一、创建RSLinx通讯: 1.1、【Communicati…

vue2项目从0搭建(三):配置环境变量及对应的webpack配置

前言 实际业务开发中,一个项目很可能会同时配置好几套环境。 比如:常规开发环境,开发测试环境,正式的测试环境,预发测试环境,客户甲的生产环境,客户乙的生产环境,通用生产环境,独立应用环境,微前端环境,大屏专用环境,移动端环境。 一女多嫁的实际业务场景,就需要我们进行多样…

mac上Homebrew的安装与使用

打开终端:command空格 ,搜索‘’终端 ’,打开终端 在终端中输入以下命令并按下回车键: /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"这个命令会自动下载并安装…

某60区块链安全之未初始化的存储指针实战二学习记录

系列文章目录 文章目录 系列文章目录未初始化的存储指针实战二实验目的实验环境实验工具实验原理实验内容实验过程EXP利用 未初始化的存储指针实战二 实验目的 学会使用python3的web3模块 学会分析以太坊智能合约未初始化的存储指针漏洞 找到合约漏洞进行分析并形成利用 实验…

win11渗透武器库,囊括所有渗透工具

开箱即用,最全的武器库,且都是2023年11月最新版,后续自己还可以再添加,下载地址:https://download.csdn.net/download/weixin_59679023/88565739 服务连接 信息收集工具 端口扫描 代理抓包 漏洞扫描 指纹识别 webshel…

apipost接口200状态码,浏览器控制台500状态码

后端 url 登录login方法 login(){this.$refs.loginForm.validate(async valid > {if (!valid) return// 由于data属性是一个json对象,需要进行解构赋值{data:result},进行状态码判断const {data: result} await this.$http.post(/api/doLogin,this.…

解决在Windows10或Windows11下无权限修改hosts文件

解决在Windows10或Windows11下无权限修改hosts文件,无法写入内容 1、首先在开始菜单中找到这个 2、接着输入: C:\Windows\System32\drivers\etc3、再次输入以下命令行:notepad hosts ,并回车: notepad hosts 4、然后…

Java专题(二)反射

反射概述 Java 反射就是在运行的状态下,对于任意一个类,都能知道这个类的任意的属性和方法,并且能调用这些方法或者改变这些类的属性,因此 Java 被称为准动态语言。 动态语言和静态语言 上面说了 Java 是准动态的语言&#xff0c…

面向对象三大特性,类与接口,java重写与重载,对象相等的判断, hashCode 与 equals

文章目录 2.1 面向对象三大特性2.1.1 封装 继承 多态2.1.2 其中Java 面向对象编程三大特性:封装 继承 多态2.1.3 关于继承如下 3 点请记住:2.1.4 什么是多态机制?Java语言是如何实现多态的?2.1.5 Java实现多态有三个必要条件&…

电脑中提示关于ntdll.dll错误怎么办,解决出现ntdll.dll错误的办法

ntdll.dll是Windows操作系统的一个关键系统文件,它包含了许多核心函数和系统调用,对于系统的稳定运行至关重要。然而,有时我们可能会遇到ntdll.dll报错的问题,导致程序无法正常运行。那么今天就和大家谈谈电脑中提示关于ntdll.dll…

企业计算机服务器中了360勒索病毒怎么办,360勒索病毒解密文件恢复

计算机技术的不断发展,为企业的生产运营提供了极大便利,不仅提升了办公效率,还促进了企业的发展。企业计算机在日常工作中一定加以防护,减少网络威胁事件的产生,确保企业的生产生产运营。最近,网络上的360后…

两巨头Facebook 和 GitHub 联手推出 Atom-IDE

9月13日,GitHub 宣布与 Facebook 合作推出了 Atom-IDE —— 它包括一系列将类 IDE 功能带到 Atom 的可选工具包。初次发布的版本包括更智能、感知上下文的自动完成;导航功能,如大纲视图和定义跳转(outline view and goto-definition)&#xf…

爬虫项目实战:利用基于selenium框架的爬虫模板爬取豆瓣电影Top250

👋 Hi, I’m 货又星👀 I’m interested in …🌱 I’m currently learning …💞 I’m looking to collaborate on …📫 How to reach me … README 目录(持续更新中) 各种错误处理、爬虫实战及模…

【JAVA】SpringBoot + mongodb 分页、排序、动态多条件查询及事务处理

【JAVA】SpringBoot mongodb 分页、排序、动态多条件查询及事务处理 1.引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><!-- mongodb ↓ -->&…