news 2026/4/15 2:45:12

STM32 EXTI(外部中断)详解

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
STM32 EXTI(外部中断)详解

一、EXTI是什么?

EXTI = External Interrupt/Event Controller(外部中断/事件控制器)

简单说:EXTI是STM32的“门卫”,专门管引脚上的电平变化。当引脚电平变化时,EXTI会通知CPU:“有情况!”

二、EXTI能干什么?

两种工作模式:

  1. 中断模式:电平变化 → EXTI → CPU → 执行中断函数

  2. 事件模式:电平变化 → EXTI → 直接触发其他外设(不打扰CPU)

常见用途:

  • 按键检测(按一下触发一次)

  • 限位开关(碰到就停)

  • 外部信号计数(来一个脉冲记一次)

三、EXTI的硬件连接(重要!)

外部引脚 → GPIO端口 → EXTI线路 → NVIC → CPU ↑ ↑ ↑ ↑ ↑ 电平变化 端口选择 线路选择 优先级 执行中断

关键规则:

  1. 每个EXTI线路对应多个GPIO引脚(但有规矩!)

  2. 同一时间,一个EXTI线路只能连一个GPIO引脚

四、EXTI线路分配表(必须记住!)

EXTI线路可以连接的GPIO引脚
EXTI0PA0, PB0, PC0, PD0, PE0, ... PI0
EXTI1PA1, PB1, PC1, PD1, PE1, ... PI1
EXTI2PA2, PB2, PC2, PD2, PE2, ... PI2
......(按数字对应)
EXTI15PA15, PB15, PC15, PD15, ... PI15

重要规律:EXTI0可以连所有端口的引脚0,EXTI1连所有引脚1,以此类推。

五、完整配置步骤(按键为例)

第1步:初始化按键引脚(PA0为例)

void KEY_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 1. 开启GPIOA时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); // 2. 配置PA0为上拉输入(按键按下接地) GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN; // 输入模式 GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; // 上拉电阻 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); }

第2步:配置EXTI中断(核心!)

void EXTI0_Init(void) { EXTI_InitTypeDef EXTI_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; // 1. 开启SYSCFG时钟(必须!很多人忘记) RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE); // 2. 将PA0连接到EXTI0线路 SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOA, EXTI_PinSource0); // 3. 配置EXTI0线路 EXTI_InitStructure.EXTI_Line = EXTI_Line0; // 线路0 EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; // 中断模式 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; // 下降沿触发 EXTI_InitStructure.EXTI_LineCmd = ENABLE; // 使能 EXTI_Init(&EXTI_InitStructure); // 4. 配置NVIC中断优先级 NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn; // 中断源 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); }

第3步:编写中断服务函数

// EXTI0中断服务函数(函数名固定,不能改!) void EXTI0_IRQHandler(void) { // 1. 检查是否是EXTI0触发的中断 if(EXTI_GetITStatus(EXTI_Line0) != RESET) { // 2. 你的处理代码 if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == 0) { // 按键按下的处理 LED_Toggle(); // 切换LED状态 } // 3. 清除中断标志(必须!不然会一直进中断) EXTI_ClearITPendingBit(EXTI_Line0); } }

六、触发方式详解

触发方式说明适用场景
上升沿触发低电平→高电平时触发按键释放检测
下降沿触发高电平→低电平时触发按键按下检测
双边沿触发上升和下降都触发电平变化检测
// 选择触发方式 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; // 上升沿 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; // 下降沿 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising_Falling; // 双边沿

七、多按键配置示例(多个EXTI线路)

// 配置PA0(EXTI0)和PC13(EXTI13)两个按键 void EXTI_Multi_Init(void) { // ... 开启时钟、GPIO初始化 ... // PA0 -> EXTI0(按键1) SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOA, EXTI_PinSource0); EXTI_InitStructure.EXTI_Line = EXTI_Line0; EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; EXTI_Init(&EXTI_InitStructure); // PC13 -> EXTI13(按键2) SYSCFG_EXTILineConfig(EXTI_PortSourceGPIOC, EXTI_PinSource13); EXTI_InitStructure.EXTI_Line = EXTI_Line13; EXTI_Init(&EXTI_InitStructure); // 配置两个中断通道 NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn; NVIC_Init(&NVIC_InitStructure); NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn; // 注意!10-15共用 NVIC_Init(&NVIC_InitStructure); } // 中断服务函数 void EXTI0_IRQHandler(void) // 处理EXTI0 { if(EXTI_GetITStatus(EXTI_Line0) != RESET) { // 处理按键1 EXTI_ClearITPendingBit(EXTI_Line0); } } void EXTI15_10_IRQHandler(void) // 处理EXTI10-15 { if(EXTI_GetITStatus(EXTI_Line13) != RESET) { // 处理按键2(PC13) EXTI_ClearITPendingBit(EXTI_Line13); } }

八、注意事项(避坑指南)

常见错误1:忘记开SYSCFG时钟

// 必须加! RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG, ENABLE); 常见错误2:中断函数名写错
  • EXTI0-4:EXTI0_IRQHandlerEXTI1_IRQHandler...

  • EXTI5-9:共用EXTI9_5_IRQHandler

  • EXTI10-15:共用EXTI15_10_IRQHandler

常见错误3:没清除中断标志

会导致重复进入中断,程序卡死!

常见错误4:GPIO模式设错

必须设置为输入模式:GPIO_Mode_IN

九、事件模式 vs 中断模式

// 中断模式:需要CPU参与 EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; // 事件模式:直接触发其他外设(如DMA、ADC) EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Event; // 事件模式不触发中断,不需要NVIC配置

十、软件触发中断(特殊用途)

// 有时候需要软件模拟一个中断 EXTI_GenerateSWInterrupt(EXTI_Line0); // 软件触发EXTI0中断

十一、消抖处理(实际应用)

void EXTI0_IRQHandler(void) { if(EXTI_GetITStatus(EXTI_Line0) != RESET) { // 简单延时消抖 delay_ms(20); // 延时20ms // 再次检测 if(GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == 0) { // 确实按下了 LED_Toggle(); } EXTI_ClearITPendingBit(EXTI_Line0); } }

十二、EXTI配置速查表

步骤做什么关键函数/配置
1开SYSCFG时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_SYSCFG)
2GPIO初始化模式=输入,上下拉按需
3引脚连EXTISYSCFG_EXTILineConfig(端口源, 引脚源)
4配置EXTI选线路、模式、触发方式
5配置NVIC设优先级
6写中断函数函数名固定,记得清标志

总结口诀

EXTI是门卫,管脚电平变。 先开时钟后连线,SYSCFG别忘记。 引脚线路要对齐,0对0来1对1。 触发方式有三种,上升下降和双边。 中断函数名固定,清除标志要牢记。 按键记得要消抖,实际应用才稳定。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/7 14:55:30

【Matlab】evalin( ‘base‘,‘var1‘)中的base是什么意思?

目录 evalin(base,var1)中base的详细解释 📌 一句话总结 🏗️ 工作区架构图示 🔍 详细解释 1. 基础工作区的特点 2. 为什么需要evalin(base, ...) 📊 evalin的完整用法 语法格式 应用示例 在MATLAB中,evalin函数用于在指定工作区中执行MATLAB表达式。 这里的…

作者头像 李华
网站建设 2026/4/13 11:51:59

全球代理ip是什么?一文搞懂原理与应用

在当前数字化时代,全球代理IP正成为企业开展数据采集与跨境业务的关键技术工具。无论是对海外市场进行调研,还是获取公开数据支持业务决策,企业都离不开稳定、快速的全球IP资源支持。那么,全球代理IP的工作原理是什么?…

作者头像 李华
网站建设 2026/4/11 8:39:05

告别音乐平台限制:Spotube跨平台音乐播放器深度使用指南

告别音乐平台限制:Spotube跨平台音乐播放器深度使用指南 【免费下载链接】spotube spotube - 一个开源、跨平台的 Spotify 客户端,使用 Spotify 的数据 API 和 YouTube 作为音频源,适合希望在不同平台上使用 Spotify 服务的开发者。 项目地…

作者头像 李华
网站建设 2026/4/11 11:10:11

精通Maestro移动UI自动化:实战无障碍测试与高级技巧解析

精通Maestro移动UI自动化:实战无障碍测试与高级技巧解析 【免费下载链接】maestro Painless Mobile UI Automation 项目地址: https://gitcode.com/gh_mirrors/ma/maestro 在移动应用开发日益复杂的今天,确保应用界面对所有用户友好且可访问已成为…

作者头像 李华
网站建设 2026/4/13 3:41:13

Golang棋牌游戏后端实战

在实时互动类游戏开发中,棋牌游戏因其强状态依赖、高频率交互和严格公平性要求,成为后端系统设计的“试金石”。而 Go 语言(Golang)凭借其轻量级并发模型、高效网络处理能力和简洁的工程结构,成为构建高性能棋牌后端的…

作者头像 李华