[stm32]stm32按键外部中断,如何消抖

本文探讨了STM32微控制器下按键外部中断的有效消抖方法,通过具体代码示例,介绍了如何在中断处理中避免按键抖动,确保输入信号的稳定性。文章对比了使用外部中断与轮询方式的不同,提供了详细的实现步骤。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

讨论一下stm32下按键外部中断如何进行有效的消抖

工程的代码是直接使用stm32 cubeMX进行配置生成的,下面就一起讨论吧。
1. 在中断处理服务函数中注释清除中断标志的语句,防止按键抖动而不断的进入中断服务程序中
2. 进入回调函数后先进性延时,一般为10ms,进行消抖,然后再判断引脚的电平状态
3. 在结束时一定注意要延时一段时间,然后才清除中断标志,而且要相对消抖时间要长一些,目的是为了松开按键时产生的抖动而又再次进入中断服务程序中,产生的抖动
4. 其实第3步是极其不严谨的。延时一段时间再清除中断标志,如果在清除中断标志的时候。按键仍未松开呢,等到松开时候,仍会有抖动。正确的应该是等待引脚电平的释放,eg(1 != HAL_GPIO_ReadPin(WK_UP_GPIO_Port,WK_UP_Pin)) 。引脚释放了才清除相应的中断标志

注:该处理方式仍属于阻塞等待的方式哦,小伙伴们可以把阻塞等待方式改为使用定时器去定时检测的方式

void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{
  /* EXTI line interrupt detected */
  if(__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != RESET)
  {
//    __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin);
    HAL_GPIO_EXTI_Callback(GPIO_Pin);
  }
}

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){
	if(WK_UP_Pin == GPIO_Pin){
        HAL_Delay(10);
        if(1 == HAL_GPIO_ReadPin(WK_UP_GPIO_Port,WK_UP_Pin));
            myprintf(&huart1, "wake up key pressed\n");
	}
	else if(KEY0_Pin == GPIO_Pin){
        HAL_Delay(10);
        if(0 == HAL_GPIO_ReadPin(KEY0_GPIO_Port,KEY0_Pin))
            myprintf(&huart1, "key 0 pressed\n");
	}
	else if(KEY1_Pin == GPIO_Pin){
        HAL_Delay(10);
        if(0 == HAL_GPIO_ReadPin(KEY1_GPIO_Port,KEY1_Pin))
            myprintf(&huart1, "key 1 pressed\n");
	}
	else if(KEY2_Pin == GPIO_Pin){
        HAL_Delay(10);
        if(0 == HAL_GPIO_ReadPin(KEY2_GPIO_Port,KEY2_Pin))
            myprintf(&huart1, "key 2 pressed\n");
    }
	HAL_Delay(100);
    __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin);
}

如图:消抖成功
在这里插入图片描述

下面是之前使用标准库写的,程序采用时间片轮询(周期5ms)的方式进行任务处理,不用外部中断的轮询方式

void Key_Config(void){
	GPIO_InitTypeDef GPIO_InitStructure;
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC,ENABLE);
	// Key1 / Key2 / Key3 -> IPU
	GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;
	GPIO_InitStructure.GPIO_Pin=GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;	
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	// SW1 -> IPU
	GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IPU;		//SW°´¼üÉÏÀ­ÊäÈë
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
	GPIO_Init(GPIOB,&GPIO_InitStructure);
	// SW2 -> IPU
	GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IPU;		//SW°´¼üÉÏÀ­ÊäÈë
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
}

u8 Key_Scan(void){
	u8 i,temp;
	for(i=0;i<5;i++)	{		
		temp=KEY_VAL>>i;	
		if(0x1f!=KEY_VAL && 0==(temp & 0x01))
			return i+1;
	}	
	return 0;	
}
u8 Get_Key(void){
	static u8 n,status=0,last_val;
	u8 key_val;
	key_val=Key_Scan();	
	if(0!=key_val && key_val==last_val){
		if(0==status)	status=1;						//°´¼ü°´Ï±êʶ
		else 	n++;
	}
	if(1==status && key_val==0){
		status=0;													//°´¼üÊͷűêʶ
		if(n>2 && n<25){	//40ms ~ 500ms		//¶Ì°´µ¥»÷ʱ¼ä¿ØÖÆ£¬¸ù¾Ý¿ØÖÆÖÜÆÚʱ¼ä¿ØÖÆ´ÎÊý
			n=0;														//Çå³ý¼ÆÊý
			return last_val;								//·µ»Ø¼üÖµ
		}
		else if(n>=25){		//´óÓÚ 500ms			//³¤°´Ê±¼ä¿ØÖÆ
			n=0;														//Çå³ý¼ÆÊý
			return last_val+10;							//³¤°´¼üÖµ = (¶Ì°´¼üÖµ+10)
		}
		else{ 															//°´¼üʱ¼ä¹ý¶Ì±»ÈÏΪÊǰ´¼ü¶¶¶¯»òÎó°´×´Ì¬
			n=0;														//Çå³ý¼ÆÊý
			return 20;											//·µ»ØÎó°´×´Ì¬
		}
	}
	last_val=key_val;										//±£´æÉÏÒ»´ÎµÄ¼üÖµ
	return 0;
}

void TIM2_Config(void){
	TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
	/*	Tout=((arr+1)*(psc+1))/Tclk	*/
	TIM_TimeBaseStructure.TIM_Prescaler=719;         				//Ô¤·ÖƵ£º72 000 000 / (719+1) = 1 000 00 Hz
  TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up; //¼ÆÊý·½Ê½£ºÏòÉϼÆÊý
  TIM_TimeBaseStructure.TIM_Period= 1999;            				//¼ÆÊýÖÜÆÚ£º1 000 00 / (4999 + 1) = 50 Hz = 0.02 s = 20 ms
  TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;     //ʱÖÓ²»·Ö¸î	
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseStructure);
	
	NVIC_Config(0,0,TIM2_IRQn,2);	
	TIM_ITConfig(TIM2,TIM_IT_Update , ENABLE);	
	TIM_Cmd(TIM2,ENABLE);	
}

void TIM2_IRQHandler(void){	
	if(TIM_GetFlagStatus(TIM2,TIM_IT_Update)!=RESET)	{
		Key_Val=Get_Key();				//ÔÚÖжϳÌÐòÖлñÈ¡¼üÖµ
		DATA_Task();							//ͨ¹ý°´¼ü»ñȡʹÄÜÄǸöÊý¾Ý´«Ê书ÄÜ£¬Êý¾Ý´«Ê亯ÊýÔÚÖ÷º¯ÊýÖе÷ÓÃ
		TIM_ClearFlag(TIM2,TIM_IT_Update);	
	}
}

小伙伴们有什么更好的方式吗,欢迎下方评论

为了设计一款能够有效聚合校园生活信息的APP,首先需要从用户的需求出发,详细分析目标用户群体的特点和需求,即XXXX生的日常生活和学习需求。在此基础上,定义APP的核心功能和服务范围,包括但不限于选课指南、考试资源、兼职信息、生活充值等服务。 参考资源链接:[校园生活APP:服务XXXX生的创新创业计划](https://wenku.csdn.net/doc/7ji9n8gsq2?spm=1055.2569.3001.10343) 接下来,进行市场调研,了解同类APP的现状、优缺点以及目标用户对此类APP的使用体验反馈。这一步骤将帮助我们确定产品的差异化特征,并据此制定出创新点和改进方向。 根据调研结果,设计APP的架构和用户界面。架构设计要考虑到数据的高效采集、存储、处理和展示,以及第三方平台的接口对接。用户界面设计则需要注重用户体验,确保操作简便、界面友好。 确定技术路线后,开始APP的开发工作。开发过程中,选择合适的开发工具和框架,如React Native或Flutter等跨平台开发框架,可以快速构建iOS和安卓双端应用。对于数据聚合,可以利用爬虫技术定时从校园网站、合作伙伴网站等来源抓取信息,并存储于服务器。开发中还应考虑到数据的安全性和隐私保护措施。 对于信息的更新及时性与准确性,可以设立后台管理系统,由专业团队负责日常的信息维护和更新,同时设置用户反馈机制,鼓励用户上报错误信息或提出建议。利用机器学习等技术优化信息分类和匹配准确性。 测试阶段,进行全面的系统测试、性能测试和用户测试,确保APP的稳定性和易用性。在APP上线后,进行持续的监控和迭代更新,根据用户反馈不断优化功能和提升服务质量。 总结来说,开发一款校园生活APP需要深入理解用户需求,合理规划产品功能,采用合适的开发技术和架构,并且注重后期的运营和维护。为了深入了解如何将这些步骤具体实现,建议参考《校园生活APP:服务XXXX生的创新创业计划》。这份文档不仅提供了产品规划的全面视角,还包括市场分析和运营策略,是解决当前问题的有力支持。 参考资源链接:[校园生活APP:服务XXXX生的创新创业计划](https://wenku.csdn.net/doc/7ji9n8gsq2?spm=1055.2569.3001.10343)
评论 21
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值