stm32中按键长短按实现方法

 主函数中扫描该函数即可,需要注意按键按下时程序会停留在该函数中,知道松手或时间到0.8s,大于0.8秒为长按,小于为短按,可以自己根据需要修改,初始化函数因板子而异,这里就不列出来了

u8 KEY_N; //在头文件中声明一下外部变量,用于主函数中判断哪个按键按下
u8 KEY_Scan(void) //返回值temp代表按键按下的时间(=temp*10 MS)
{
	static u8 flag=1;
	u8 temp=0;
	if(flag==1)
	{
	  if(KEY1_IN==0) //下面四个按键相似操作
		{
			flag=0;
			while(KEY1_IN==0&&temp<85)
			{
				temp++;
				Delay_Ms(10);
			}
			KEY_N=1;
			return temp;
		}
		
		if(KEY2_IN==0) 
		{
			flag=0;
			while(KEY1_IN==0&&temp<85)
			{
				temp++;
				Delay_Ms(10);
			}
			KEY_N=2;
			return temp;
		}
		
		if(KEY3_IN==0) 
		{
			flag=0;
			while(KEY1_IN==0&&temp<85)
			{
				temp++;
				Delay_Ms(10);
			}
			KEY_N=3;
			return temp;
		}
		
		if(KEY4_IN==0) 
		{
			flag=0;
			while(KEY1_IN==0&&temp<85)
			{
				temp++;
				Delay_Ms(10);
			}
			KEY_N=4;
			return temp;
		}
	}
	if(KEY1_IN==1&&KEY2_IN==1&&KEY3_IN==1&&KEY4_IN==1)
	{
		flag=1;
		KEY_N=0;
	}
		return 0;
}

 还有就是怎么利用上面这个扫描函数实现按键长按时某个量连续增减,下面以时钟的小时位长按连加为例:

从上面扫描函数代码中可以看到当按键按下时间到0.85s程序就会返回了,也就是主程序会往下执行了,但是KEY_N还不为0,直到按键松开后才会变为0,所以在主函数中就可以通过判断这个变量来判断按键是否松开,未松开则每隔50ms自增一次,延时不能太短,不然眼睛看不出来。

while(KEY_N)
{
    Time_Hour++;
    if(Time_Hour==24) Time_Hour=0;
																	        
    sprintf((char*)LCD_String,"    %d:%d:%d   ",Time_Hour,Time_Min,Time_Sec);
																	    
    LCD_DisplayStringLine(Line3 ,LCD_String);
    Delay_Ms(50);
    KEY_Scan();
}

 

  • 3
    点赞
  • 42
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
首先需要了解一下STM32F103C8T6是一款基于ARM Cortex-M3内核的微控制器,它拥有丰富的外设和强大的处理能力。实现长短按键代码的关键在于采用定时器断的方式来检测按键的状态,同时还需要使用GPIO外设来配置按键的输入输出。下面是一个简单的示例代码,实现长短按键的检测: ```c #include "stm32f10x.h" #define LONG_PRESS_TIME 500 // 定义按时间为500ms GPIO_InitTypeDef GPIO_InitStructure; TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; NVIC_InitTypeDef NVIC_InitStructure; uint8_t buttonState = 0; // 按钮状态(0:未按下,1:按下) void TIM2_IRQHandler(void) // 定时器断处理函数 { if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) // 判断是否为定时器更新断 { if (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == 0) // 判断是否为按键按下状态 { if (buttonState == 0) // 判断之前是否已经检测到按键按下 { TIM_Cmd(TIM2, DISABLE); // 禁止定时器计数 TIM_SetCounter(TIM2, 0); // 计数器清零 buttonState = 1; // 设置按钮状态为按下 } else { if (TIM_GetCounter(TIM2) >= LONG_PRESS_TIME) // 判断是否为按 { // 按处理 } else { // 短按处理 } buttonState = 0; // 设置按钮状态为未按下 } } TIM_ClearITPendingBit(TIM2, TIM_IT_Update); // 清除定时器更新断标志位 } } int main(void) { RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); // 使能定时器2时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // 使能GPIOA时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; // 配置PA0为输入模式 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; GPIO_Init(GPIOA, &GPIO_InitStructure); NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn; // 配置定时器2断通道 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); TIM_TimeBaseStructure.TIM_Period = 1000; // 设置计数器自动重装值为1000,即1ms计数一次 TIM_TimeBaseStructure.TIM_Prescaler = 7199; // 设置预分频值为7199,即计数频率为72MHz/7200=10kHz TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE); // 使能定时器2更新断 TIM_Cmd(TIM2, ENABLE); // 启动定时器2 while (1) { // 循环等待断触发,不做其他操作 } } ``` 以上代码仅供参考,具体实现方式还需根据具体需求进行调整。以下是相关问题:

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值