正文

【013】查询方式按键 [51]2006-06-25 00:57:00

【评论】 【打印】 【字体: 】 本文链接:http://blog.pfan.cn/wentao/16138.html

分享到:

实验参考: 笨笨工作室-实验六、查询方式按键。(查看)
实验板:   FB51A。(查看)

实验目的: <1> 掌握读取外部按键时候应该注意的事项,掌握查询方式响应按键的方法。
     <2> 掌握读取按键去抖动的方法。
      <3> 掌握判直接寻址位转移指令jb,jnb的用法。

实验现象: 最右面的数码管初始显示0,每次按下外部按键K1执行加一,计数到9时重新回0。


单片机响应外部按键的方式有两种,一种是查询方式,一种是中断方式。

查询方式:单片机不断的查询是否有按键按下,如果有按键按下的话,就执行相应的程序,否则继续查询。

中断方式:单片机处理自己的工作,如果有按键按下,向单片机发出中断请求。单片机停下现在正在处理的工作,转去执行中断程序,执行之后回来继续刚才的工作。

本实验是用查询方式完成, 只用到一个键K1,每次按下,使第一位数码管加1显示,计数到9里重新回0。显示部分电路及查表显示方法参考:【005】数码管显示数字 和 【006】多位数码管动态显示。原理图如下:


由原理图知:当按键k1按下的时候,对应P1.0输入低电平。因此,我们要查询k1是否按下,只要检测到p1.0是低电平即可。 在查询键是否按下时,要注意按键的去抖动问题,关于键的消抖,见 【012】按键消抖


按上述方法编写程序:

        org   0000h
start:  mov   p0, #48h       ; 段码-显示0
        mov   p2, #01h       ; 位码-选通第一位数码管

main:   mov   r7, #0ffh      ; 查表时偏移量,用来实现由0到9循环显示
key:    jb    p1.0, key      ; 如果P1.0为高电平,则继续执行此句查询。
        call  del10ms        ; 否则调用延时程序实现软件去抖
        jb    p1.0, key      ; 再次查询P1.0,如果是高电平,表示是抖动,继续查询
       
dis:                         ; 确定按键按下,开始执行显示程序
        inc   r7             ; 偏移量自加(第一次值为0)
        mov   a, r7
        mov   dptr, #tab     ; 取表首址
        movc  a, @a+dptr     ; 用查表指令获取要显示的数字
        mov   p0, a          ; 得到该字段码送入P0口显示
        cjne  a, #48h, key   ; 未显示到最后一个数, 继续查询按键状态
        ljmp  main           ; 显示到最后一个数字则重新开始
       
tab:    db    0ebh, 52h, 62h, 0e1h, 64h, 44h, 0eah, 40h, 60h, 48h ; 段码表

del10ms:                     ; 10ms延时子程序(12M)
         mov  r6, #20        ; 2机器周期
temp:    mov  r5, #248       ; 2机器周期
         djnz r5, $          ; 2机器周期 2+2×248=498
         djnz r6, temp       ; 2机器周期 2×20=40
         ret                 ; 2+20×498+40=10002 即10ms
end

将该程序下载到板上验证发现很不稳定,按一次键时显示可能会变几次,如果按住键不放,将会显示“8”,且不停在闪烁,用Porteus仿真了一下也是这个问题。这是由于显示了一个数字之后,单片机会返回重新查询P1.0的状态,如果按下的时间很长,就会导致单片机又检测到低电平,所以就继续显示下一个数字,常按住的话就会循环此过程,所以看到的闪烁的“8”实际上是快速动态显示“0”到“9”时人眼的视觉暂留产生的。那如何使按下一次按键,无论时间长短,只作一次按键处理呢?可以在显示完一个数字后即检测P1.0是否回到高电平(键放开),若是,表示按键一次,继续执行显示下一个数; 若不是则原地等待直到P1.0返回高电平。按此思路在程序中将一个段码送入P0口显示之后,加一判断语句(红色):

        org   0000h
start:  mov   p0, #48h       ; 段码-显示0
        mov   p2, #01h       ; 位码-选通第一位数码管

main:   mov   r7, #0ffh      ; 查表时偏移量,用来实现由0到9循环显示
key:    jb    p1.0, key      ; 如果P1.0为高电平,则继续执行此句查询。
        call  del10ms        ; 否则调用延时程序实现软件去抖
        jb    p1.0, key      ; 再次查询P1.0,如果是高电平,表示是抖动,继续查询
       
dis:                         ; 确定按键按下,开始执行显示程序
        inc   r7             ; 偏移量自加(第一次值为0)
        mov   a, r7
        mov   dptr, #tab     ; 取表首址
        movc  a, @a+dptr     ; 用查表指令获取要显示的数字
        mov   p0, a          ; 得到该字段码送入P0口显示
wait2:  jnb   p1.0, wait2    ; 等待按键松开
        cjne  a, #48h, key   ; 未显示到最后一个数, 继续查询按键状态
        ljmp  main           ; 显示到最后一个数字则重新开始
       
tab:    db    0ebh, 52h, 62h, 0e1h, 64h, 44h, 0eah, 40h, 60h, 48h ; 段码表

del10ms:                     ; 10ms延时子程序(12M)
         mov  r6, #20        ; 2机器周期
temp:    mov  r5, #248       ; 2机器周期
         djnz r5, $          ; 2机器周期 2+2×248=498
         djnz r6, temp       ; 2机器周期 2×20=40
         ret                 ; 2+20×498+40=10002 即10ms
end

再下载到板上验证,可得到预测结果,用Porteus仿真如下


阅读(5524) | 评论(4)


版权声明:编程爱好者网站为此博客服务提供商,如本文牵涉到版权问题,编程爱好者网站不承担相关责任,如有版权问题请直接与本文作者联系解决。谢谢!

评论

loading...
您需要登录后才能评论,请 登录 或者 注册