
你有没有想过一个问题:用GPIO读取按键,CPU要不断循环检查电平(轮询),效率太低。有没有不占用CPU的方式,让按键一按下就通知CPU?**有。EXTI(外部中断/事件控制器)就是干这个的。**它硬件检测GPIO的跳变沿,检测到就触发中断。
**那个"边沿检测"的电路(对应教程6.1节),**EXTI内部有三个关键寄存器:
- 上升沿触发选择寄存器(RTSR):写1使能上升沿触发
- 下降沿触发选择寄存器(FTSR):写1使能下降沿触发
- 软件中断事件寄存器(SWIER):写1软件触发中断(用于调试)
同时使能RTSR和FTSR的同一个通道 → 双边沿触发。配置示例:
EXTI->RTSR |= (1 << 0); // EXTI0上升沿触发
EXTI->FTSR |= (1 << 0); // EXTI0下降沿触发
那个"挂起寄存器"的清除(对应教程5.12节), 中断触发后,EXTI_PR(挂起寄存器)对应位被硬件置1。NVIC看到挂起位=1,就去执行中断处理函数。关键一步:中断处理函数结束前,必须清除挂起位,否则会反复进中断。

EXTI->PR = (1 << 0); // 写1清除挂起位
很多初学者忘这一步,死活找不出为什么中断一直触发。
那个"线共享"的烦恼(对应教程5.12节), EXTI0线同时连接了PA0、PB0、PC0、PD0。多个引脚共享同一条中断线,无法从EXTI本身区分是哪个引脚触发的。设计PCB时:需要中断的引脚,分散到不同的EXTI线(例如PA0、PA1、PA2、PA3),不要全部挤在PA0、PB0、PC0上。如果确实无法避开,就在中断处理函数里读取各个引脚的电平,软件判断。
那个"软件中断"的调试(对应教程6.1节), 调试时,没有硬件信号,想测试中断处理函数怎么办?EXTI支持软件触发中断:写SWIER寄存器对应位。
EXTI->SWIER |= (1 << 0); // 软件触发EXTI0中断
硬件检测到SWIER的写入,立即将挂起位置1。不需要按键,不需要信号发生器,一行代码让中断跑起来。

这个故事的启示, 为什么需要EXTI?因为CPU不应该浪费时间去"看"信号 。信号来了,硬件通知CPU;没来,CPU做自己的事。轮询是主动看,中断是被动等。被动等更省力。
写在最后, 下次你要检测GPIO电平变化,别用while循环读引脚。想想EXTI。上升沿、下降沿、双边沿,硬件自动捕捉。信号来了叫醒我,没来别烦我。
(本文灵感源于于振南《新概念ARM32单片机》教程第6.1节"EXTI按键中断实验与定时器",感谢作者将EXTI边沿检测的精髓讲得如此通透。)
如果您觉得这个故事对您有启发,欢迎点赞、转发,让更多工程师看到这个藏在EXTI背后的"被动等待"智慧。
