【GPIO8个函数解释】

函数解释

  1. void GPIO_DeInit(GPIO_TypeDef* GPIOx);
  • 作用:将指定GPIO端口的所有寄存器恢复为默认值。这会清除之前对该端口的所有配置,使其回到初始状态。
  • 使用方法:传入要复位的GPIO端口指针,例如GPIOAGPIOB等。
  1. void GPIO_AFIODeInit(void);
  • 作用:将复用功能I/O(AFIO)的寄存器恢复为默认值。复用功能I/O用于将一些外设功能映射到特定的GPIO引脚上,该函数可清除这些映射配置。
  • 使用方法:无需传入参数,直接调用即可。
  1. void GPIO_Init(GPIO_TypeDef* GPIOx, GPIO_InitTypeDef* GPIO_InitStruct);
  • 作用:根据GPIO_InitStruct结构体中的参数初始化指定的GPIO端口。该结构体包含了GPIO引脚的模式、速度等配置信息。
  • 使用方法:传入要初始化的GPIO端口指针和一个GPIO_InitTypeDef结构体指针。
  1. void GPIO_StructInit(GPIO_InitTypeDef* GPIO_InitStruct);
  • 作用:将GPIO_InitTypeDef结构体的成员初始化为默认值。通常在使用GPIO_Init函数之前调用该函数,以确保结构体中的成员有合理的初始值。
  • 使用方法:传入一个GPIO_InitTypeDef结构体指针。

示例代码

以下是一个简单的示例,展示了如何使用这些函数来初始化GPIO端口:

#include "stm32f10x.h"int main(void)
{GPIO_InitTypeDef GPIO_InitStructure;// 使能GPIOA时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);// 将GPIOA的所有寄存器恢复为默认值GPIO_DeInit(GPIOA);// 将复用功能I/O的寄存器恢复为默认值GPIO_AFIODeInit();// 将GPIO_InitStructure结构体的成员初始化为默认值GPIO_StructInit(&GPIO_InitStructure);// 配置GPIOA的引脚0为推挽输出模式,最大速度为50MHzGPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;// 根据GPIO_InitStructure结构体的参数初始化GPIOAGPIO_Init(GPIOA, &GPIO_InitStructure);while (1){// 点亮LEDGPIO_SetBits(GPIOA, GPIO_Pin_0);// 延时一段时间for (int i = 0; i < 1000000; i++);// 熄灭LEDGPIO_ResetBits(GPIOA, GPIO_Pin_0);// 延时一段时间for (int i = 0; i < 1000000; i++);}
}

代码解释

  1. 使能时钟:在使用GPIO之前,需要先使能相应的时钟。这里使用RCC_APB2PeriphClockCmd函数使能GPIOA的时钟。
  2. 复位寄存器:调用GPIO_DeInitGPIO_AFIODeInit函数将GPIOA和复用功能I/O的寄存器恢复为默认值。
  3. 初始化结构体:调用GPIO_StructInit函数将GPIO_InitStructure结构体的成员初始化为默认值。
  4. 配置GPIO:设置GPIO_InitStructure结构体的成员,指定要配置的引脚、模式和速度。
  5. 初始化GPIO端口:调用GPIO_Init函数根据GPIO_InitStructure结构体的参数初始化GPIOA。
  6. 循环控制LED:在主循环中,使用GPIO_SetBitsGPIO_ResetBits函数点亮和熄灭连接到GPIOA引脚0的LED。

通过以上步骤,你可以使用这些函数来配置和控制STM32的GPIO端口。

函数名功能参数说明适用场景
GPIO_SetBits将指定GPIO端口的指定引脚置为高电平(逻辑1)- GPIOx:指定的GPIO端口,如GPIOAGPIOB
- GPIO_Pin:指定的引脚,如GPIO_Pin_0GPIO_Pin_1
仅需将引脚拉高的场景,操作简单直接
GPIO_ResetBits将指定GPIO端口的指定引脚置为低电平(逻辑0)- GPIOx:指定的GPIO端口,如GPIOAGPIOB
- GPIO_Pin:指定的引脚,如GPIO_Pin_0GPIO_Pin_1
仅需将引脚拉低的场景,操作简单直接
GPIO_WriteBit将指定GPIO端口的指定引脚设置为指定的电平值- GPIOx:指定的GPIO端口,如GPIOAGPIOB
- GPIO_Pin:指定的引脚,如GPIO_Pin_0GPIO_Pin_1
- BitVal:电平值,Bit_SET表示高电平,Bit_RESET表示低电平
需要灵活设置引脚高低电平的场景
GPIO_Write将一个16位的值写入指定的GPIO端口,一次性设置该端口所有引脚的电平状态- GPIOx:指定的GPIO端口,如GPIOAGPIOB
- PortVal:一个16位的值,用于设置端口所有引脚的电平
需要对一个端口的多个引脚进行批量操作的场景

这四个函数均为STM32标准库中用于操作GPIO引脚输出电平的函数,下面为你对比分析它们的区别,并给出使用示例。

函数区别分析

1. void GPIO_SetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);
  • 功能:将指定GPIO端口的指定引脚置为高电平(逻辑1)。
  • 特点:只能用于将引脚置高,操作简单直接,适用于只需要将引脚拉高的场景。
2. void GPIO_ResetBits(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin);
  • 功能:将指定GPIO端口的指定引脚置为低电平(逻辑0)。
  • 特点:只能用于将引脚置低,操作简单直接,适用于只需要将引脚拉低的场景。
3. void GPIO_WriteBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, BitAction BitVal);
  • 功能:将指定GPIO端口的指定引脚设置为指定的电平值,电平值可以是高电平(Bit_SET)或低电平(Bit_RESET)。
  • 特点:可以灵活设置引脚的高低电平,通过传入不同的BitAction参数来控制。
4. void GPIO_Write(GPIO_TypeDef* GPIOx, uint16_t PortVal);
  • 功能:将一个16位的值写入指定的GPIO端口,一次性设置该端口所有引脚的电平状态。
  • 特点:可以同时控制一个端口的所有16个引脚,适用于需要对多个引脚进行批量操作的场景。

示例代码

以下是使用这四个函数的示例代码:

#include "stm32f10x.h"int main(void)
{GPIO_InitTypeDef GPIO_InitStructure;// 使能GPIOA时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);// 配置GPIOA的引脚0和引脚1为推挽输出模式,最大速度为50MHzGPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);while (1){// 使用GPIO_SetBits将引脚0置高GPIO_SetBits(GPIOA, GPIO_Pin_0);// 延时一段时间for (int i = 0; i < 1000000; i++);// 使用GPIO_ResetBits将引脚0置低GPIO_ResetBits(GPIOA, GPIO_Pin_0);// 延时一段时间for (int i = 0; i < 1000000; i++);// 使用GPIO_WriteBit将引脚1置高GPIO_WriteBit(GPIOA, GPIO_Pin_1, Bit_SET);// 延时一段时间for (int i = 0; i < 1000000; i++);// 使用GPIO_WriteBit将引脚1置低GPIO_WriteBit(GPIOA, GPIO_Pin_1, Bit_RESET);// 延时一段时间for (int i = 0; i < 1000000; i++);// 使用GPIO_Write同时设置引脚0和引脚1的电平// 0x03 表示二进制 0000 0000 0000 0011,即引脚0和引脚1都置高GPIO_Write(GPIOA, 0x03);// 延时一段时间for (int i = 0; i < 1000000; i++);// 使用GPIO_Write同时设置引脚0和引脚1的电平// 0x00 表示二进制 0000 0000 0000 0000,即引脚0和引脚1都置低GPIO_Write(GPIOA, 0x00);// 延时一段时间for (int i = 0; i < 1000000; i++);}
}

代码解释

  1. 使能时钟和配置GPIO:使能GPIOA的时钟,并将GPIOA的引脚0和引脚1配置为推挽输出模式。
  2. 使用GPIO_SetBitsGPIO_ResetBits:分别将引脚0置高和置低,实现引脚0的电平翻转。
  3. 使用GPIO_WriteBit:通过传入Bit_SETBit_RESET参数,将引脚1置高和置低,实现引脚1的电平翻转。
  4. 使用GPIO_Write:通过传入不同的16位值,一次性设置引脚0和引脚1的电平状态。

以下是三种GPIO控制方法的对比及效果分析:


1. 方法对比

方法1GPIO_SetBitsGPIO_ResetBits
GPIO_SetBits(GPIOA, GPIO_Pin_0);   // 置高电平
GPIO_ResetBits(GPIOA, GPIO_Pin_0); // 置低电平
  • 原理:直接操作寄存器,通过宏定义实现。
  • 特点
    • 代码简洁,无函数调用开销,执行效率高。
    • 专用于单一操作(置高或置低),功能明确。
方法2GPIO_WriteBit 配合 Bit_RESETBit_SET
GPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_RESET); // 置低电平
GPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_SET);   // 置高电平
  • 原理:通过函数调用设置电平,参数使用枚举值(BitAction)。
  • 特点
    • 代码直观,参数明确(Bit_SET/Bit_RESET)。
    • 灵活性高,可动态指定电平状态(如通过变量控制)。
方法3GPIO_WriteBit 配合强制类型转换
GPIO_WriteBit(GPIOA, GPIO_Pin_0, (BitAction)0); // 置低电平
GPIO_WriteBit(GPIOA, GPIO_Pin_0, (BitAction)1); // 置高电平
  • 原理:强制将整数 01 转换为 BitAction 类型。
  • 特点
    • 功能上等同于方法2(BitAction 内部对应 01)。
    • 缺点:代码可读性差,潜在兼容性风险(依赖枚举底层实现)。

2. 效果是否相同?

  • 功能层面:三种方法最终均会修改GPIO输出寄存器的对应位,实现引脚电平控制,效果完全相同
  • 性能层面
    • 方法1(宏操作)直接操作寄存器,无函数调用,效率最高
    • 方法2和方法3(函数调用)有轻微函数调用开销,但实际应用中差异可忽略。

3. 代码规范建议

  1. 推荐方法2
    使用 Bit_SETBit_RESET,兼顾可读性、规范性和灵活性。

    GPIO_WriteBit(GPIOA, GPIO_Pin_0, Bit_RESET); // 正确写法
    
  2. 避免方法3
    强制类型转换会降低代码可维护性,且不符合ST库的设计意图。

  3. 方法1的适用场景
    若需极致性能(如高频信号控制),可优先选择 GPIO_SetBits/GPIO_ResetBits


4. 补充说明

  • 条件编译 #if 0 的作用
    代码中被 #if 0 包裹的部分会被编译器忽略,仅用于临时禁用代码块。
    若要启用方法1或方法2,需将对应的 #if 0 改为 #if 1

总结

三种方法在功能上完全等效,但代码风格和可维护性差异显著。推荐优先使用方法2(Bit_SET/Bit_RESET,既能保证效率,又符合工程规范。

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

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

相关文章

将图表和表格导出为PDF的功能

<template><div><divref"pdfContent"style"position: relative; width: 800px; margin: 0 auto"><!-- ECharts 图表 --><div id"chart" style"width: 100%; height: 400px" /><!-- Element UI 表格 …

C++中的链表操作

在C中&#xff0c;链表是一种常见的数据结构&#xff0c;它由一系列节点组成&#xff0c;每个节点包含数据部分和指向下一个节点的指针。C标准库&#xff08;STL&#xff09;中提供了std::list和std::forward_list两种链表实现&#xff0c;分别对应双向链表和单向链表。此外&am…

蛋白设计 ProteinMPNN

传统方法的局限性是什么&#xff1f; 传统蛋白质设计方法的局限性&#xff1a; 基于物理的传统方法&#xff0c;例如罗塞塔&#xff0c;面临计算难度&#xff0c;因为需要计算所有可能结构的能量&#xff0c;包括不需要的寡聚态和聚合态。 设计目标与显式优化之间缺乏一致性通…

有哪些开源的视频生成模型

1. 阿里巴巴通义万相2.1&#xff08;WanX 2.1&#xff09; 技术架构&#xff1a;基于Diffusion Transformer&#xff08;DiT&#xff09;架构&#xff0c;结合自研的高效变分自编码器&#xff08;VAE&#xff09;和Flow Matching训练方案&#xff0c;支持时空上下文建模。参数…

【动态规划】最长上升子序列模板

最长上升子序列 题目传送门 一、题目描述 给定一个长度为 N 的数列&#xff0c;求数值严格单调递增的子序列的长度最长是多少。 输入格式 第一行包含整数 N。 第二行包含 N 个整数&#xff0c;表示完整序列。 输出格式 输出一个整数&#xff0c;表示最大长度。 数据范围 …

LeetCode 891 -- 贡献度思想

题目描述 子序列宽度之和 思路 ref 代码 相似题 子数组范围和 acwing

化工行业如何通过定制化工作流自动化实现25-30%成本优化?

作者&#xff1a;Mihir Jhaveri 编译&#xff1a;李升伟 发布日期&#xff1a;2024年10月30日 在化工生产领域&#xff0c;数字化转型正以颠覆性态势重塑产业格局。通过集成定制化软件、ERP系统、工业物联网&#xff08;IIoT&#xff09;传感网络、机器人流程自动化&#xff0…

Compose组件转换XML布局

文章目录 学习JetPack Compose资源前言&#xff1a;预览界面的实现Compose组件的布局管理一、Row和Colum组件&#xff08;LinearLayout&#xff09;LinearLayout&#xff08;垂直方向 → Column&#xff09;LinearLayout&#xff08;水平方向 → Row&#xff09; 二、相对布局 …

RAG测试数据集资源

一、通用问答基准数据集 HotpotQA 特点:包含11万+多跳问答对最佳用途:测试复杂推理能力数据示例:{"question": "Were Scott Derrickson and Ed Wood of the same nationality?","answer": "Yes, both are American" }MS MARCO 特点…

快速掌握MCP——Spring AI MCP包教包会

最近几个月AI的发展非常快&#xff0c;各种大模型、智能体、AI名词和技术和框架层出不穷&#xff0c;作为一个业余小红书博主的我最近总刷到MCP这个关键字&#xff0c;看着有点高级我也来学习一下。 1.SpringAI与functionCall简单回顾 前几个月我曾写过两篇关于SpringAI的基础…

学习笔记--(6)

import numpy as np import matplotlib.pyplot as plt from scipy.special import erfc# 设置参数 rho 0.7798 z0 4.25 # 确保使用大写 Z0&#xff0c;与定义一致def calculate_tau(z, z_prime, rho, s_values):return np.log(rho * z * z_prime * s_values / 2)# 定义 chi_…

【AI4CODE】5 Trae 锤一个基于百度Amis的Crud应用

【AI4CODE】目录 【AI4CODE】1 Trae CN 锥安装配置与迁移 【AI4CODE】2 Trae 锤一个 To-Do-List 【AI4CODE】3 Trae 锤一个贪吃蛇的小游戏 【AI4CODE】4 Trae 锤一个数据搬运工的小应用 1 百度 Amis 简介 百度 Amis 是一个低代码前端框架&#xff0c;由百度开源。它通过 J…

认识 Promise

认识 Promise 前言&#xff1a;为什么会出现 Promise&#xff1f; 最常见的一个场景就是 ajax 请求&#xff0c;通俗来说&#xff0c;由于网速的不同&#xff0c;可能你得到返回值的时间也是不同的&#xff0c;这个时候我们就需要等待&#xff0c;结果出来了之后才知道怎么样…

纯c++实现transformer 训练+推理

项目地址 https://github.com/freelw/cpp-transformer C 实现的 Transformer 这是一个无需依赖特殊库的 Transformer 的 C 实现&#xff0c;涵盖了训练与推理功能。 本项目使用C复刻了《Dive into Deep Learning》中关于 Transformer 的第 11 章11.7小节点内容。构建了一个英…

Go 语言规范学习(7)

文章目录 Built-in functionsAppending to and copying slicesClearCloseManipulating complex numbersDeletion of map elementsLength and capacityMaking slices, maps and channelsMin and maxAllocationHandling panicsBootstrapping PackagesSource file organizationPac…

Python Cookbook-5.1 对字典排序

任务 你想对字典排序。这可能意味着需要先根据字典的键排序&#xff0c;然后再让对应值也处于同样的顺序。 解决方案 最简单的方法可以通过这样的描述来概括:先将键排序&#xff0c;然后由此选出对应值: def sortedDictValues(adict):keys adict.keys()keys.sort()return …

Git Rebase 操作中丢失提交的恢复方法

背景介绍 在团队协作中,使用 Git 进行版本控制是常见实践。然而,有时在执行 git rebase 或者其他操作后,我们可能会发现自己的提交记录"消失"了,这往往让开发者感到恐慌。本文将介绍几种在 rebase 后恢复丢失提交的方法。 问题描述 当我们执行以下操作时,可能…

C语言基础要素(019):输出ASCII码表

计算机以二进制处理信息&#xff0c;但二进制对人类并不友好。比如说我们规定用二进制值 01000001 表示字母’A’&#xff0c;显然通过键盘输入或屏幕阅读此数据而理解它为字母A&#xff0c;是比较困难的。为了有效的使用信息&#xff0c;先驱者们创建了一种称为ASCII码的交换代…

鸿蒙定位开发服务

引言 鸿蒙操作系统&#xff08;HarmonyOS&#xff09;作为面向万物互联时代的分布式操作系统&#xff0c;其定位服务&#xff08;Location Kit&#xff09;为开发者提供了多场景、高精度的位置能力支持。本文将从技术原理、开发流程到实战案例&#xff0c;全面解析鸿蒙定位服务…

rknn_convert的使用方法

rknn_convert是RKNN-Toolkit2提供的一套常用模型转换工具&#xff0c;通过封装上述API接口&#xff0c;用户只需编辑模型对应的yml配置文件&#xff0c;就可以通过指令转换模型。以下是如何使用rknn_convert工具的示例命令以及支持的指令参数&#xff1a; python -m rknn.api.…