首页 > 其他分享 >【STM32】IO口取反 | 寄存器方式 | 异或运算符 | 原理

【STM32】IO口取反 | 寄存器方式 | 异或运算符 | 原理

时间:2024-08-09 13:27:05浏览次数:23  
标签:STM32 Pin PIN 引脚 取反 运算符 InitStructure GPIO

LuckiBit

目录

STM32 IO口取反 | 寄存器方式 | 异或运算符 | 原理

1. 引言

在嵌入式系统中,IO口(输入/输出口)的控制是非常基础且重要的操作。STM32作为一种广泛使用的微控制器,提供了多种方式来操作其IO口。其中,利用寄存器直接操作和使用异或运算符来取反IO口的状态是一种高效且灵活的方法。本文将详细介绍如何通过寄存器方式和异或运算符对STM32的GPIO引脚进行取反操作,帮助读者深入理解这一过程。

2. GPIO基础知识

2.1 GPIO概述

GPIO(General Purpose Input/Output),即通用输入/输出,是微控制器中一种可以被配置为输入或输出状态的数字信号管脚。通过GPIO,可以实现对外部设备的控制,如LED、按钮、传感器等。

2.2 STM32的GPIO架构

STM32的GPIO架构具有以下特点:

  • 每个GPIO引脚可以配置为输入或输出模式。
  • 支持多种工作模式,如推挽输出、开漏输出、浮空输入、上拉输入、下拉输入等。
  • 提供多种速度选择,以满足不同应用的需求。
  • 通过寄存器操作可以实现对GPIO的高效控制。

2.3 GPIO寄存器简介

STM32的GPIO控制通过一系列寄存器实现,主要包括以下几个:

  • GPIOx_CRL/CRH(配置寄存器低/高):配置引脚的模式和功能。
  • GPIOx_IDR(输入数据寄存器):读取引脚的输入状态。
  • GPIOx_ODR(输出数据寄存器):设置引脚的输出状态。
  • GPIOx_BSRR(位设置/复位寄存器):原子性地设置和复位引脚。
  • GPIOx_BRR(位复位寄存器):复位引脚。
  • GPIOx_LCKR(配置锁定寄存器):锁定引脚的配置。

3. GPIO引脚取反原理

GPIO引脚取反是指将当前引脚的电平状态进行翻转,即高电平变低电平,低电平变高电平。通过寄存器操作和异或运算符可以高效实现这一功能。

3.1 寄存器操作实现取反

GPIO引脚的输出状态存储在ODR(输出数据寄存器)中,通过对该寄存器进行操作,可以实现引脚状态的修改。具体操作步骤如下:

  1. 读取当前ODR寄存器的值。
  2. 对需要取反的引脚位进行异或操作。
  3. 将修改后的值写回ODR寄存器。

3.2 异或运算符的应用

异或运算符(^)是一种位运算符,对于任意位a和b,a ^ b的结果如下:

  • 若a和b相同,则结果为0。
  • 若a和b不同,则结果为1。

利用这一特性,可以通过与1进行异或操作实现位的取反。例如,若某引脚对应的位为1,与1进行异或操作后结果为0;若对应位为0,与1进行异或操作后结果为1,从而实现取反效果。

4. 示例代码

4.1 基础示例:LED闪烁

以下代码演示了如何通过寄存器方式和异或运算符实现LED灯的闪烁:

#include "stm32f10x.h"        // 包含 STM32 的设备头文件
#include "stm32f10x_gpio.h"   // 包含 STM32 的 GPIO 头文件

void GPIO_Config(void) {
    // 使能 GPIOA 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    // 配置 GPIOA 引脚 5 为推挽输出模式
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_5;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) {
    // 使用异或运算符取反 GPIO 引脚
    GPIOx->ODR ^= GPIO_Pin;
}

int main(void) {
    // 配置 GPIO
    GPIO_Config();
    
    while (1) {
        // 取反 GPIOA 引脚 5
        GPIO_TogglePin(GPIOA, GPIO_PIN_5);
        
        // 简单的延时
        for (volatile int i = 0; i < 1000000; i++);
    }
}

4.2 应用实例:继电器控制

继电器控制是嵌入式系统中的常见应用之一,通过类似的方法可以实现继电器的开关控制。

#include "stm32f10x.h"        // 包含 STM32 的设备头文件
#include "stm32f10x_gpio.h"   // 包含 STM32 的 GPIO 头文件

void GPIO_Config(void) {
    // 使能 GPIOB 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    
    // 配置 GPIOB 引脚 12 为推挽输出模式
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_12;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
}

void GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) {
    // 使用异或运算符取反 GPIO 引脚
    GPIOx->ODR ^= GPIO_Pin;
}

int main(void) {
    // 配置 GPIO
    GPIO_Config();
    
    while (1) {
        // 取反 GPIOB 引脚 12
        GPIO_TogglePin(GPIOB, GPIO_PIN_12);
        
        // 简单的延时
        for (volatile int i = 0; i < 1000000; i++);
    }
}

5. GPIO引脚配置详解

5.1 GPIO_InitTypeDef结构体

在STM32的标准外设库中,GPIO的初始化通过GPIO_InitTypeDef结构体实现。该结构体包含以下成员:

  • GPIO_Pin:指定要配置的GPIO引脚,可以是GPIO_Pin_x的组合。
  • GPIO_Speed:指定引脚的输出速度。
  • GPIO_Mode:指定引脚的工作模式。

5.2 配置示例

以下代码展示了如何配置GPIOA的引脚5为推挽输出模式,速度为50MHz:

GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Pin = GPIO_PIN_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);

6. GPIO寄存器操作详解

6.1 输出数据寄存器(GPIOx_ODR)

ODR寄存器用于控制GPIO引脚的输出状态,每个引脚对应一个位。例如,要设置GPIOA的引脚5为高电平,可以通过以下操作实现:

GPIOA->ODR |= GPIO_PIN_5;  // 将引脚5置为高电平

要将引脚5置为低电平,可以使用以下操作:

GPIOA->ODR &= ~GPIO_PIN_5;  // 将引脚5置为低电平

6.2 位设置/复位寄存器(GPIOx_BSRR)

BSRR寄存器提供了一种原子性设置和复位引脚的方式,通过向BSRR寄存器写入相应的位,可以同时设置和复位不同的引脚。例如,要设置引脚5为高电平,同时复位引脚6,可以使用以下操作:

GPIOA->BSRR = GPIO_PIN_5 | (GPIO_PIN_6 << 16);

7. 异或运算符的应用详解

7.1 异或运算符基础

异或运算符(^)是一种位运算符,用于对两个二进制数的对应位进行异或操作。其运算规则如下:

  • 若两个对应位相同,则结果为0。
  • 若两个对应位不同,则结果为1。

例如:

  • 0 ^ 0 = 0
  • 0 ^ 1 = 1
  • 1 ^ 0 = 1
  • 1 ^ 1 = 0

7.2 实现GPIO引脚取反

利用异或运算符可以方便地实现GPIO引脚的取反操作。假设当前引脚状态为1,要将其取反,即将其状态变为0,可以通过与1进行异或操作实现。同理,当前引脚状态为0,通过与1进行异或操作后状态变为1,从而实现状态的取反。

7.3 取反操作示例

以下代码展示了如何通过异或运算符对GPIO引脚进行取反操作:

#include "stm32f10x.h"        // 包含 STM32 的设备头文件
#include "stm32f10x_gpio.h"   // 包含 STM32 的 GPIO 头文件

void GPIO_Config(void) {
    // 使能 GPIOA 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    // 配置 GPIOA 引脚 5 为推挽输出模式
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_5;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) {
    // 使用异或运算符取反 GPIO 引脚
    GPIOx->ODR ^= GPIO_Pin;
}

int main(void) {
    // 配置 GPIO
    GPIO_Config();
    
    while (1) {
        // 取反 GPIOA 引脚 5
        GPIO_TogglePin(GPIOA, GPIO_PIN_5);
        
        // 简单的延时
        for (volatile int i = 0; i < 1000000; i++);
    }
}

7.4 复杂应用示例:多引脚取反

以下代码展示了如何对多个GPIO引脚同时进行取反操作:

#include "stm32f10x.h"        // 包含 STM32 的设备头文件
#include "stm32f10x_gpio.h"   // 包含 STM32 的 GPIO 头文件

void GPIO_Config(void) {
    // 使能 GPIOA 时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    // 配置 GPIOA 引脚 5 和 引脚 6 为推挽输出模式
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_5 | GPIO_PIN_6;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void GPIO_TogglePins(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pins) {
    // 使用异或运算符取反 GPIO 引脚
    GPIOx->ODR ^= GPIO_Pins;
}

int main(void) {
    // 配置 GPIO
    GPIO_Config();
    
    while (1) {
        // 取反 GPIOA 引脚 5 和 引脚 6
        GPIO_TogglePins(GPIOA, GPIO_PIN_5 | GPIO_PIN_6);
        
        // 简单的延时
        for (volatile int i = 0; i < 1000000; i++);
    }
}

8. 应用场景

8.1 LED控制

通过上述方法,可以方便地控制LED灯的状态。例如,可以在按钮按下时取反LED灯的状态,达到开关LED灯的效果。

#include "stm32f10x.h"
#include "stm32f10x_gpio.h"

void GPIO_Config(void) {
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_5;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);
}

void Button_Config(void) {
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_13;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU; // 上拉输入
    GPIO_Init(GPIOC, &GPIO_InitStructure);
}

int main(void) {
    GPIO_Config();
    Button_Config();
    
    while (1) {
        if (GPIO_ReadInputDataBit(GPIOC, GPIO_PIN_13) == 0) { // 按钮按下
            GPIO_TogglePin(GPIOA, GPIO_PIN_5);
            // 消抖延时
            for (volatile int i = 0; i < 100000; i++);
            while (GPIO_ReadInputDataBit(GPIOC, GPIO_PIN_13) == 0); // 等待按钮松开
        }
    }
}

8.2 继电器控制

在工业控制中,继电器控制是常见应用。通过GPIO引脚的取反操作,可以实现继电器的开关控制。

#include "stm32f10x.h"
#include "stm32f10x_gpio.h"

void GPIO_Config(void) {
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    
    GPIO_InitTypeDef GPIO_InitStructure;
    GPIO_InitStructure.GPIO_Pin = GPIO_PIN_12;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
}

void Relay_Toggle(void) {
    GPIOB->ODR ^= GPIO_PIN_12;
}

int main(void) {
    GPIO_Config();
    
    while (1) {
        Relay_Toggle();
        for (volatile int i = 0; i < 1000000; i++);
    }
}

9. 总结

本文详细介绍了通过寄存器方式和异或运算符对STM32的GPIO引脚进行取反操作的方法。通过这些方法,可以高效、灵活地控制GPIO引脚的状态,适用于多种嵌入式应用场景。以下是本文的要点总结:

  • GPIO引脚的基本概念和STM32的GPIO架构。
  • GPIO寄存器的功能和使用方法。
  • 异或运算符的基本原理及其在GPIO引脚取反中的应用。
  • 通过具体示例演示如何实现GPIO引脚的取反操作。

通过掌握这些内容,可以更好地理解和应用STM32的GPIO控制,为嵌入式系统开发打下坚实的基础。

附录:常用GPIO宏定义

#define GPIO_PIN_0  ((uint16_t)0x0001)  /*!< Pin 0 selected */
#define GPIO_PIN_1  ((uint16_t)0x0002)  /*!< Pin 1 selected */
#define GPIO_PIN_2  ((uint16_t)0x0004)  /*!< Pin 2 selected */
#define GPIO_PIN_3  ((uint16_t)0x0008)  /*!< Pin 3 selected */
#define GPIO_PIN_4  ((uint16_t)0x0010)  /*!< Pin 4 selected */
#define GPIO_PIN_5  ((uint16_t)0x0020)  /*!< Pin 5 selected */
#define GPIO_PIN_6  ((uint16_t)0x0040)  /*!< Pin 6 selected */
#define GPIO_PIN_7  ((uint16_t)0x0080)  /*!< Pin 7 selected */
#define GPIO_PIN_8  ((uint16_t)0x0100)  /*!< Pin 8 selected */
#define GPIO_PIN_9  ((uint16_t)0x0200)  /*!< Pin 9 selected */
#define GPIO_PIN_10 ((uint16_t)0x0400)  /*!< Pin 10 selected */
#define GPIO_PIN_11 ((uint16_t)0x0800)  /*!< Pin 11 selected */
#define GPIO_PIN_12 ((uint16_t)0x1000)  /*!< Pin 12 selected */
#define GPIO_PIN_13 ((uint16_t)0x2000)  /*!< Pin 13 selected */
#define GPIO_PIN_14 ((uint16_t)0x4000)  /*!< Pin 14 selected */
#define GPIO_PIN_15 ((uint16_t)0x8000)  /*!< Pin 15 selected */
#define GPIO_PIN_ALL ((uint16_t)0xFFFF) /*!< All pins selected */

10. 结束语

  1. 本节内容已经全部介绍完毕,希望通过这篇文章,大家对 STM32IO 口取反有了更深入的理解和认识。
  2. 感谢各位的阅读和支持,如果觉得这篇文章对你有帮助,请不要吝惜你的点赞和评论,这对我们非常重要。再次感谢大家的关注和支持点我关注❤️

相关文章:

标签:STM32,Pin,PIN,引脚,取反,运算符,InitStructure,GPIO
From: https://blog.csdn.net/EleganceJiaBao/article/details/140985840

相关文章

  • STM32之SPI
    ADI对SPI的介绍1.硬件结构普通并联结构(使用片选信号CS或者NSS选通芯片)菊花链结构(部分芯片支持)2.通讯方式(四种)3.通信速率......
  • JAVA修饰符、运算符、循环语句
    JAVA修饰符修饰符用来定义类、方法或者变量,通常放在语句的最前端一、访问修饰符1、default默认访问修饰符:在同一包内可见,不使用任何修饰符,使用对象为类、接口、变量、方法,访问级别是包级别(package-level),即只能被同一包中的其他类访问2、private私有访问修饰符:最严格的访问级......
  • C++ 禁用类的拷贝构造函数和赋值运算符
    C++中如果没有显式定义类的构造函数和赋值运算符,编译器会自动生成对应的函数,但是对于一些含有指针成员变量的类,自动生成的成员函数只会进行浅拷贝,会导致动态申请的内存在对象析构的时候doublefree,引起崩溃的问题。因此如果没有必要,通常会禁用该接口,避免用户调用该接口造成问题。......
  • STM32CubleMX创建FreeRtos工程教程,图文教程
        前言:STM32CubeMX是一个开发工具,它已经将FreeRTOS这个实时操作系统(RTOS)集成到其工具中。换句话说,通过STM32CubeMX,可以非常方便地为STM32微控制器生成配置代码,其中包括对FreeRTOS的支持。    而本篇就是使用STM32CubleMX,生成支持FreeRtos的图文教程......
  • 基于STM32开发的智能灌溉系统
    目录引言环境准备工作硬件准备软件安装与配置系统设计系统架构硬件连接代码实现初始化代码控制代码应用场景农业灌溉园艺灌溉常见问题及解决方案常见问题解决方案结论1.引言智能灌溉系统通过监测土壤湿度和环境条件,自动控制水泵和阀门,实现精确灌溉,从而......
  • 基于STM32开发的智能门禁系统
    目录引言环境准备工作硬件准备软件安装与配置系统设计系统架构硬件连接代码实现初始化代码控制代码应用场景公司门禁管理家庭门禁管理常见问题及解决方案常见问题解决方案结论1.引言智能门禁系统通过RFID卡或密码输入的方式,实现对门禁的智能控制和管理......
  • 1392、STM32单片机温湿度检测阈值报警4个继电器加4个负载风扇等无线蓝牙远程(程序+原
    毕设帮助、开题指导、技术解答(有偿)见文未 目录方案选择单片机的选择显示器选择方案一、设计功能二、实物图三、原理图四、程序源码五、PCB图资料包括:需要完整的资料可以点击下面的名片加下我,找我要资源压缩包的百度网盘下载地址及提取码。方案选择单片机的选......
  • 运算符续集
    1.扩展增值运算符**省略了一些加减删除的步骤而直接运算的运算符**2.面试题区分""+a+b和a+b+""3.条件运算符(必须掌握)如果符合条件x则输出y,否则输出z......
  • STM32之IIC协议
    物理层 1.从机数量选择地址限制:IIC协议本身没有严格规定总线上device最大数目,从理论上看,IIC能挂的device数目取决于能表示的最大地址空间,在7位地址模式下,减去0x00地址不可用,理论上可以挂2^7-1=127个设备。总线电容限制:由于器件的管脚都是有输入电容的,PCB上......
  • 从 python 设置运算符符号到方法名称的映射是什么?它们与文档不匹配
    我创建了自己的Customset类,它实现了python集合的几乎所有方法。当我使用此自定义集的实例时,许多集合运算符都会失败。它们会失败,并显示类似以下内容的内容:TypeError:unsupportedoperandtype(s)for-:'Customset'andCustomset'orTypeError:'<='......