stm32学习笔记:中断的应用:对射式红外传感器计次旋转编码器计次

本文主要是介绍stm32学习笔记:中断的应用:对射式红外传感器计次旋转编码器计次,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

相关API介绍

EXT配置API(stm32f10x exti.h)

NVIC 配置API (misc.h)

 初始化的中断的步骤

    第一步:配置RCC时钟,把涉及外设的时钟都打开
    第二步:配置GPIO,设置为输入模式
    第三步:配置AFIO,选择某个GPIO口连接到EXTI(边缘检测及控制器)
    第四步:配置EXTI(不需要开启时钟,原因不详),选择边沿触发方式和触发响应方式
    边沿触发方式:上升沿、下降沿、或者双边沿,触发响应方式:中断响应和事件响应
    第五步:配置NVIC(内核的外设,不需要开启时钟),给中断选择一个合适的优先级

 

 对射式红外传感器计次

CountSensor.h

#ifndef __COUNT_SENEOR_H
#define __COUNT_SENEOR_Hextern void CountSensor_Init(void);
extern uint16_t CountSensor_Get(void);#endif
CountSensor.c
#include "stm32f10x.h"                  // Device headeruint16_t CountSensor_Count;void CountSensor_Init(void)
{//第一步:配置RCC时钟,把涉及外设的时钟都打开//第二步:配置GPIO,设置为输入模式//第三步:配置AFIO,选择某个GPIO口连接到EXTI(边缘检测及控制器)//第四步:配置EXTI(不需要开启时钟,原因不详),选择边沿触发方式和触发响应方式//边沿触发方式:上升沿、下降沿、或者双边沿,触发响应方式:中断响应和事件响应//第五步:配置NVIC(内核的外设,不需要开启时钟),给中断选择一个合适的优先级//对射式红外传感器:B14RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);GPIO_InitTypeDef GPIO_InitStructure;//参考手册中文版中,第八章GPIO有说明说明外设设置什么格式GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;//上拉输入,默认为高电平GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOB, &GPIO_InitStructure);//配置AFIO的数据选择器,选择想要的中断引脚GPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource14);EXTI_InitTypeDef EXTI_InitStructure;EXTI_InitStructure.EXTI_Line = EXTI_Line14;EXTI_InitStructure.EXTI_LineCmd = ENABLE;EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;//中断模式(而不是事件响应)//三种,上升沿,下降沿,上升沿+下降沿EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;//下降沿触发EXTI_Init(&EXTI_InitStructure);//5种分组方式选择其中的一种NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);NVIC_InitTypeDef NVIC_InitStructure;//在stm32f10x.h中选择,该芯片是MD中等密度的,锁选择STM32F10X_MD即可//stm32的EXTI10到EXTI15都是合并到EXTI15_10_IRQn通道的NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn;//指定通道NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//在misc文件中查找NVIC_Priority_Table,查的分组2的抢占优先级和响应优先级的取值范围均为0-3NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;NVIC_Init(&NVIC_InitStructure);
}uint16_t CountSensor_Get(void)
{return CountSensor_Count;//全局变量
}//中断函数不需要申明,因为不需要调用,是直接申明的
//中断函数都是无参,无返回值
void EXTI15_10_IRQHandler(void)	//中断函数的名字都是固定的
{//因为10-15通道都可以进来,故要判断是不是想要的14通道进来if (EXTI_GetITStatus(EXTI_Line14) == SET){/*如果出现数据乱跳的现象,可再次判断引脚电平,以避免抖动*/if (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_14) == 0){CountSensor_Count ++;}//中断程序结束后,一定要再调用一下清楚中断标志位的函数,//只有中断标志位置1,程序就会跳转到中断函数//如果不清除中断标志位,就会一直申请中断,//这样程序就会不断响应中断,执行中断函数,程序就会卡死在中断函数中EXTI_ClearITPendingBit(EXTI_Line14);}
}

 main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "CountSensor.h"int main(void)
{OLED_Init();CountSensor_Init();//从第一行,第一列开始写OLED_ShowString(1, 1, "Count:");while (1){//从第一行第7列开始写OLED_ShowNum(1, 7, CountSensor_Get(), 5);}
}

 旋转编码器计次

Encoder.c

如果把A相的下降沿用作触发中断,在中断时刻读取B相的电平
则正转是高电平,反转是低电平
正转时,A相先出现下降沿,所以刚开始动,就进中断了
反转后,A相先出现下降沿,转到位了,才进入中断

故该实验:A、B相都触发中断
正转:B相下降沿、A相低电平
反转:A相下降沿、B相低电平
这样保证正转和反转都是转到位,才执行数字加减的操作

#include "stm32f10x.h"                  // Device header
#include "Delay.h"int16_t EncoderCount = 0;void Encoder_Init(void)
{// 配置RCC, 将涉及到的外设全部打开, 不打开时钟外设无法工作// EXTI 和 NVIC 的时钟是一直打开的, 不需要再开启时钟了, 因为 NVIC 是内核的外设RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);    // 开启 GPIOB 的时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);     // 开启 AFIO  的时钟// 配置 GPIO_Pin_B0 | GPIO_Pin_B1, 设置端口为输入模式GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;    // 上拉、下拉、浮空输入均可GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure);// 配置 AFIO, 选择我们用的 GPIO 到后面的 EXTIGPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource0);    // PB0 -> EXTI0GPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource1);    // PB1 -> EXTI1// 配置 EXTI: 将EXTI的第0、1个线路配置为中断模式、下降沿触发、然后开启中断EXTI_InitTypeDef EXTI_InitStruct;EXTI_InitStruct.EXTI_Line = EXTI_Line0 | EXTI_Line1;    // 选择PB0 所在的第0个线路、选择 PB1 所在的第1个线路EXTI_InitStruct.EXTI_LineCmd = ENABLE;                  // 开启中断EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;        // 中断模式 或 事件模式EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Falling;    // 下降沿触发                     EXTI_Init(&EXTI_InitStruct);// 配置 NVIC, 给 EXTI0 中断选择一个合适的优先级, 最终外部中断信号就能进入 CPU 了NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);         // 2位抢占, 2位响应, 整个工程需要设置成一致的.NVIC_InitTypeDef NVIC_InitStruct;NVIC_InitStruct.NVIC_IRQChannel = EXTI0_IRQn;       // 选择进入 NVIC 的通道 NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;            // 通道使能NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 1;  // 抢占优先级: 0~3NVIC_InitStruct.NVIC_IRQChannelSubPriority = 1;         // 响应优先级: 0~3NVIC_Init(&NVIC_InitStruct);// 配置 NVIC, 给 中断选择一个合适的优先级, 最终外部中断信号就能进入 CPU 了NVIC_InitStruct.NVIC_IRQChannel = EXTI1_IRQn;NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;            // 通道使能NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 1;  // 抢占优先级: 0~3NVIC_InitStruct.NVIC_IRQChannelSubPriority = 2;         // 响应优先级: 0~3NVIC_Init(&NVIC_InitStruct);
}void EXTI0_IRQHandler(void)    
{ITStatus res = EXTI_GetITStatus(EXTI_Line0);           // 判断是否是 EXIT0if(SET == res) {// 抖动延时下 Delay_ms(3);if(GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_0) == 0 && GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0)  // 正转{EncoderCount--;}EXTI_ClearITPendingBit(EXTI_Line0);                // 清除中断标志位}	
}void EXTI1_IRQHandler(void)    
{ITStatus res = EXTI_GetITStatus(EXTI_Line1);           // 判断是否是 EXIT1if(SET == res) {// 抖动延时下 Delay_ms(3);if(GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0 && GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_0) == 0)  // 反转{EncoderCount++;}EXTI_ClearITPendingBit(EXTI_Line1);                // 清除中断标志位}	
}int16_t Encoder_Get(void)
{int16_t Temp = EncoderCount;EncoderCount = 0;return Temp;
}

 Encoder.h

#ifndef __ENCODER_H
#define __ENCODER_Hextern void Encoder_Init(void);
extern int16_t Encoder_Get(void);#endif

main.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "oled.h"
#include "encoder.h"static int16_t Num = 0;int main(void)
{				OLED_Init();Encoder_Init();OLED_ShowString(1, 1, "Num:");while(1)	{Num += Encoder_Get();OLED_ShowSignedNum(1, 5, Num, 5);}
}

这篇关于stm32学习笔记:中断的应用:对射式红外传感器计次旋转编码器计次的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/199974

相关文章

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

Node.js 中 http 模块的深度剖析与实战应用小结

《Node.js中http模块的深度剖析与实战应用小结》本文详细介绍了Node.js中的http模块,从创建HTTP服务器、处理请求与响应,到获取请求参数,每个环节都通过代码示例进行解析,旨在帮... 目录Node.js 中 http 模块的深度剖析与实战应用一、引言二、创建 HTTP 服务器:基石搭建(一

java中VO PO DTO POJO BO DO对象的应用场景及使用方式

《java中VOPODTOPOJOBODO对象的应用场景及使用方式》文章介绍了Java开发中常用的几种对象类型及其应用场景,包括VO、PO、DTO、POJO、BO和DO等,并通过示例说明了它... 目录Java中VO PO DTO POJO BO DO对象的应用VO (View Object) - 视图对象

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2

正则表达式高级应用与性能优化记录

《正则表达式高级应用与性能优化记录》本文介绍了正则表达式的高级应用和性能优化技巧,包括文本拆分、合并、XML/HTML解析、数据分析、以及性能优化方法,通过这些技巧,可以更高效地利用正则表达式进行复杂... 目录第6章:正则表达式的高级应用6.1 模式匹配与文本处理6.1.1 文本拆分6.1.2 文本合并6

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit

Qt QWidget实现图片旋转动画

《QtQWidget实现图片旋转动画》这篇文章主要为大家详细介绍了如何使用了Qt和QWidget实现图片旋转动画效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、效果展示二、源码分享本例程通过QGraphicsView实现svg格式图片旋转。.hpjavascript

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用