按键的消抖,可用硬件或软件两种方法。
1、硬件消抖
在键数较少时可用硬件方法消除键抖动。硬件消抖的典型做法是:采用R-S触发器或RC积分电路。
双稳态消抖
电路的工作过程如下:当按键未按下时,a=0,b=1,输出A=1,B=0。当按键按下时,按键的机械d性作用使按键产生前沿抖动。当释放按键时,按键的机械d性作用使按键产生后沿抖动。
滤羡燃波消抖
电路的工作过程如下:当按键未按下时,电容C两端的电压为V,非门输出为1。当按键按下时,由于电容C两端的电压不能突变,因此即使在接触过程中出现抖动,只要C两端的充电电压波动不超过非门的开启电压(TTL为0.8V左右),非门的输出就不会配态改变(可通过选取合适的R1、R2和C的值来实现)。
当按键断开时,即使出现抖动,由于C两端的电压不能突变(它要经过R2放电),因此只要C两端的放电电压波动不超过非门的关闭电压,非门的输出就不会改变所以,RC电路滤波消抖成败的关键在于R1、R2和C时间常数的选取。
必须保证C由稳态电压充电到开启电压或放电到关闭电压的延迟时间大于或等于10ms。参数的数值可由计算或实验确定,图中的参数仅供参考。若釆用输入端有施密特触发特性的门电路,则效果更好。
2、软件消抖
实现方法:假设未按键时输入1,按键后输入为0,抖动时不定。可以做以下检测:检测到按键输入为0之后,延时5ms~10ms,再次检测,如果按键还为0,那么就认为有按键输入。
延时的5ms~10ms恰好避开了抖兄卖虚动期,从而消除了前沿抖动的影响同理,在检测到按键释放后,再延时5~10ms,消除后沿抖动,然后再对键值进行处理。不过一般情况下,我们通常不对按键释放的后沿进行处理,实践证明,这样也能满足一定的要求。
扩展资料
在机械按键的触点闭合和断开时,都会产生抖动,为了保证系统能正确识别按键的开关,就必须对按键的抖动进行处理。
按键的抖动对于人类来说是感觉不到的,但对单片机来说,则是完全可以感应到的,而且还是一个很“漫长”的过程,因为单片机处理的速度在“微秒”级,而按键抖动的时间至少在“毫秒”级。
单片机如果在触点抖动期间检测按键的通断状态,则可能导致判断出错,即按键一次按下或释放被错误地认为是多次 *** 作,从而引起误处理。因此,为了确保单片机对一次按键动作只作—次响应,就必须考虑如何消除按键抖动的影响。
参考资料来源:百度百科-按键消抖
提供困则思路,仅供参考
我觉得你这种情况,应该在程序里做一个自锁程序。流程如下:
1、声明一个按键状态变量key_status,并初始化为0
2、当按键按下并抬起时(此处应该做软件滤波),认为按键被触发,此时:
如果key_status==0, 就让key_status赋值为1
如果key_status==1, 就让key_status赋值为0
3、 *** 纵杆作用七段数码管动作程序只有在key_status==1的时候执行,其他不动作
while(1){
...
if(key_status == 1)
{
孝巧 //执行 *** 纵杆作用七段数码管动作程汪慎棚序
}
...
}
#include <reg51.h>#include <intrins.h>
unsigned char data dis_digit
unsigned char key_s, key_v
unsigned char code dis_code[11]={0xc0,0xf9,0xa4,0xb0, // 0, 1, 2, 3
0x99,0x92,0x82,0xf8,0x80,0x90, 0xff}// 4, 5, 6, 7, 8, 9, off
unsigned char dis_buf[8] // 显示缓冲区
unsigned char sec_bcd[8] // 秒计数值, BCD码
unsigned char dis_index //
unsigned char key_times // K1 按下次数//
void clr_time()
void update_disbuf()
bit scan_key()
void proc_key()
void delayms(unsigned char ms)
sbit K1 = P1^0
void main(void)
{
P0 = 0xff
P3 = 0xff
TMOD = 0x11 // 定时器0, 1工作模式1, 16位定时方式
TH1 = 0xdc
TL1 = 0
TH0 = 0xFC
TL0 = 0x17
clr_time() //
dis_digit = 0x7f // 初始显示P30口数码管
dis_index = 0 //
key_times = 0
key_v = 0x01
IE = 0x8a // 使能timer0, timer1中断
TR0 = 1
TR1 = 0
while(1)
{
if(scan_key())
{
delayms(10)
if(scan_key())
{
key_v = key_s
proc_key()
}
}
}
}
void clr_time()
{
sec_bcd[0] = 0x0
sec_bcd[1] = 0x0
sec_bcd[2] = 0x0
sec_bcd[3] = 0x0
sec_bcd[4] = 0x0
sec_bcd[5] = 0x0
sec_bcd[6] = 0x0
sec_bcd[7] = 0x0
update_disbuf()
}
bit scan_key()
{
key_s = 0x00
key_s |= K1
return(key_s ^ key_v)
}
void proc_key()
{
if((key_v &0x01) == 0)
{
key_times++
if(key_times == 1)
{
TR1 = 1
}
else if(key_times == 2)
{
TR1 = 0
}
else
{
clr_time()
key_times = 0
}
}
}
void timer0() interrupt 1
// 定时器0中断服务程序, 用于数码芦歼管的动态扫描
// dis_index --- 显示索引, 用于标识当前显示的数码管和缓冲区的偏移量
// dis_digit --- 位选通值, 传送到P2口用于选通当前数码管的数值, 如等于0xfe时,
//选通P2.0口数码管
// dis_buf --- 显于缓冲区基地址
{
TH0 = 0xFC
TL0 = 0x17
P3 = 0xff //陆高 先关闭所有数码管
P0 = dis_buf[dis_index] // 显示代陪悉冲码传送到P0口
P3 = dis_digit //
dis_digit = _cror_(dis_digit,1)// 位选通值右移(P30<-P37), 下次中断时选通下一位数码管
dis_index++ //
dis_index &= 0x07 // 8个数码管全部扫描完一遍之后,再回到第一个开始下一次扫描
}
void timer1() interrupt 3
//
{
unsigned char i
TH1 |= 0xdc
for(i = 0i <8i++)
{
sec_bcd[i]++ // 低位加1
if(sec_bcd[i] <10) // 如果低位满10则向高位进1
break // 低位未满10
sec_bcd[i] = 0 // 低位满10清0
}
update_disbuf() // 更新显示缓冲区
}
void update_disbuf()
// 更新显示缓冲区
{
dis_buf[0] = dis_code[sec_bcd[0]]
dis_buf[1] = dis_code[sec_bcd[1]]
dis_buf[2] = dis_code[sec_bcd[2]] &0x7f// 加上小数点
dis_buf[3] = dis_code[sec_bcd[3]]
dis_buf[4] = dis_code[sec_bcd[4]]
dis_buf[5] = dis_code[sec_bcd[5]]
dis_buf[6] = dis_code[sec_bcd[6]]
dis_buf[7] = dis_code[sec_bcd[7]]
}
void delayms(unsigned char ms)
// 延时子程序
{
unsigned char i
while(ms--)
{
for(i = 0i <120i++)
}
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)