搜索
bottom↓
回复: 27

stm32设置外部中断,按键是否要去抖?

[复制链接]

出0入0汤圆

发表于 2015-7-13 09:54:00 | 显示全部楼层 |阅读模式
如题,我的设置是:外部中断为下降沿模式,GPIO的输入为上拉模式的输入,中断响应为输出端反转状态———小灯亮灭。当按下按键后,理应产生一个下降沿,按一下灭灯,再按一下亮灯。

但是现象是:有时候可以亮灭转换,但有时候按下去小灯状态不变

这个是什么原因,是否要进行去抖?
(初学者,用的野火的程序)

  1. //中断响应函数
  2. void EXTI9_5_IRQHandler(void)
  3. {
  4.         if(EXTI_GetITStatus(EXTI_Line5) != RESET) //确保是否产生了EXTI Line中断
  5.         {
  6.                 // LED1 取反               
  7.                 GPIO_WriteBit(GPIOC, GPIO_Pin_3,
  8.                         (BitAction)((1-GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_3))));
  9.                 EXTI_ClearITPendingBit(EXTI_Line5);     //清除中断标志位
  10.         }  
  11. }
复制代码

  1. //中断初始化设置
  2. static void NVIC_Configuration(void)
  3. {
  4.           NVIC_InitTypeDef NVIC_InitStructure;
  5.   
  6.          /* Configure one bit for preemption priority -优先级设置为第一级*/
  7.          NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
  8.        
  9.         /***********注各个端口的引脚5 中断向量一样************************/
  10.           /* 配置P[A|B|C|D|E]5为中断源 */
  11.           NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQn;
  12.           NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  13.           NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  14.           NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  15.          NVIC_Init(&NVIC_InitStructure);
  16. }
复制代码

  1. //中断函数设置
  2. void EXTI_PE5_Config(void)
  3. {
  4.         GPIO_InitTypeDef GPIO_InitStructure;
  5.         EXTI_InitTypeDef EXTI_InitStructure;

  6.         /* config the extiline(PE5) clock and AFIO clock */
  7.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE | RCC_APB2Periph_AFIO,ENABLE);
  8.                                                                                                
  9.         /* config the NVIC(PE5) */
  10.         NVIC_Configuration();

  11.         /* EXTI line gpio config(PE5) */       
  12.          GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;      
  13.          GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;         // 上拉输入
  14.          GPIO_Init(GPIOE, &GPIO_InitStructure);

  15.         /* EXTI line(PE5) mode config */
  16.         GPIO_EXTILineConfig(GPIO_PortSourceGPIOE, GPIO_PinSource5);
  17.          EXTI_InitStructure.EXTI_Line = EXTI_Line5;                                                        //外部中断线的组合,应该是引脚号
  18.           EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;                                     //一般就用这种中断模式
  19.          EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;                                         //下降沿中断
  20.          EXTI_InitStructure.EXTI_LineCmd = ENABLE;                                                        //中断使能
  21.           EXTI_Init(&EXTI_InitStructure);
  22.                                
  23. }
复制代码

阿莫论坛20周年了!感谢大家的支持与爱护!!

一只鸟敢站在脆弱的枝条上歇脚,它依仗的不是枝条不会断,而是自己有翅膀,会飞。

出0入0汤圆

 楼主| 发表于 2015-7-13 09:57:13 | 显示全部楼层
补充一下,一开始以为是按下去太快,没有检测到下降沿。但是即使慢慢的按键也会出现小灯不变的情况。

出0入4汤圆

发表于 2015-7-13 10:05:22 | 显示全部楼层
去抖是必须的。

出0入0汤圆

发表于 2015-7-13 10:06:01 | 显示全部楼层
去抖是必须的,一般都是简单的延时处理。

出0入0汤圆

 楼主| 发表于 2015-7-13 10:21:52 | 显示全部楼层
lingdianhao 发表于 2015-7-13 10:06
去抖是必须的,一般都是简单的延时处理。

但是这不是只要检测到下降沿就可以了吗?

出0入54汤圆

发表于 2015-7-13 10:28:33 | 显示全部楼层
要用CPU的速度去分析,不能用人眼看到的速度去分析。
你按一下可能会产生很多个上升沿下降沿,虽然时间很短但单片机都会响应,当检测到偶数个下降沿时变会出现灯不亮的情况。

其实一般像反转灯这种不是很紧急的事情建议用查询的方法做更可靠。

出0入0汤圆

发表于 2015-7-13 10:31:26 | 显示全部楼层
黑巧克力 发表于 2015-7-13 10:21
但是这不是只要检测到下降沿就可以了吗?

按键的电平并不是理想的下降沿,而是有毛刺的下降沿,按键信号波动的时候,可能多次进入中断,导致你的LED点亮时间太短,所以你察觉不到。一般的处理是按键IO加滤波电容,再软件延时,也就是你进入中断后,需要延时ms级后,再检测按键IO的电平是否为低,就OK了。

出0入0汤圆

 楼主| 发表于 2015-7-13 10:31:37 | 显示全部楼层
lusson 发表于 2015-7-13 10:28
要用CPU的速度去分析,不能用人眼看到的速度去分析。
你按一下可能会产生很多个上升沿下降沿,虽然时间很短 ...

说的好,

利用中断的方式看起来更简洁,用扫描的方式一直嵌套在里面循环浪费资源

而且新手学习,各个模块都尝试下。

出0入0汤圆

发表于 2015-7-13 10:32:44 | 显示全部楼层
按键按下这个过程都进了N次中断了,主要还是要分析一下按键按下的过程

出0入0汤圆

 楼主| 发表于 2015-7-13 10:35:58 | 显示全部楼层
ZY71 发表于 2015-7-13 10:32
按键按下这个过程都进了N次中断了,主要还是要分析一下按键按下的过程

这么说看来还是得在中断响应函数中,加个简单的延时啊

出0入0汤圆

 楼主| 发表于 2015-7-13 10:55:55 | 显示全部楼层
lingdianhao 发表于 2015-7-13 10:31
按键的电平并不是理想的下降沿,而是有毛刺的下降沿,按键信号波动的时候,可能多次进入中断,导致你的LE ...

嗯,可行!

加了延时之后效果还是那样,不过加了小电容之后就没问题了,这个是真没想到

谢谢各位的讨论~~~

出0入10汤圆

发表于 2015-7-13 10:57:10 | 显示全部楼层
黑巧克力 发表于 2015-7-13 10:31
说的好,

利用中断的方式看起来更简洁,用扫描的方式一直嵌套在里面循环浪费资源

只要你程序写得好,我觉得不用中断用的资源也是很少的

出0入10汤圆

发表于 2015-7-13 10:58:33 | 显示全部楼层
相反,中断我觉得利用的硬件资源才更多

出0入0汤圆

发表于 2015-7-13 10:59:10 | 显示全部楼层
黑巧克力 发表于 2015-7-13 10:35
这么说看来还是得在中断响应函数中,加个简单的延时啊

完全不建议在中断中加延时处理,当然如果MCU只做这件事或只做很少的事问题是不大。其实按键扫描的程序只要放在while中保证一定时间能扫到即可。

出0入0汤圆

发表于 2015-7-13 10:59:22 | 显示全部楼层
一般都是按下计时,释放响应的,

出0入0汤圆

发表于 2015-7-13 11:37:55 | 显示全部楼层
黑巧克力 发表于 2015-7-13 10:55
嗯,可行!

加了延时之后效果还是那样,不过加了小电容之后就没问题了,这个是真没想到

加了多大的电容?

出0入0汤圆

发表于 2015-7-13 12:41:00 | 显示全部楼层
ST的IO口就这样,我做过一个加上拉电阻后好很多。

出0入0汤圆

 楼主| 发表于 2015-7-13 13:56:22 | 显示全部楼层
modbus 发表于 2015-7-13 11:37
加了多大的电容?

随便抄了一个1000pf 的,这也有关系吗

出0入0汤圆

 楼主| 发表于 2015-7-13 13:57:15 | 显示全部楼层
hameyou 发表于 2015-7-13 11:13
放在主循环里,采用状态机扫描,连延时都省了,基本不消耗资源

放在主循环里面感觉不灵活,利用中断就不会影响后面程序了

出0入21汤圆

发表于 2015-7-13 14:03:50 | 显示全部楼层
检测按键要用中断去处理,你的程序框架绝对是差评.

出0入0汤圆

发表于 2015-7-13 14:03:59 | 显示全部楼层
按键不需要用中断处理的,除非你主循环半秒中转不过1轮

出0入0汤圆

发表于 2015-7-13 14:33:07 | 显示全部楼层
两种方式:1 主循环中查内部时间,做状态机 2 也可以中断,中断中用状态机,两种占用资源很少。

出0入0汤圆

 楼主| 发表于 2015-7-13 14:44:43 | 显示全部楼层
NJ8888 发表于 2015-7-13 14:03
按键不需要用中断处理的,除非你主循环半秒中转不过1轮

感觉用中断更加灵活一点,不需要放在while 中一直循环

出0入0汤圆

发表于 2015-7-13 16:24:49 | 显示全部楼层
你去看看论坛里边其他的按键帖子,中断里加延时是不行滴。

出0入0汤圆

发表于 2015-7-13 18:24:23 | 显示全部楼层
前几天也碰到这个问题,没怎么在意,以为是延时时间太短

出0入0汤圆

发表于 2019-4-21 11:39:48 | 显示全部楼层
非常感谢,原来关键是在硬件,而不是软件

出100入0汤圆

发表于 2019-5-7 14:32:43 | 显示全部楼层
每个口加一个1000pF电容,示波器看,上下沿就没有毛刺了,硬件能解决的问题,不交给软件
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-5-16 18:50

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

快速回复 返回顶部 返回列表