源:http://www.amobbs.com/forum.php?mod=viewthread&tid=2243715

  吸取各位前辈的经验,将之前二极管用量多的问题优化一下,目前不用二极管能接6键,2只二极管能接12键,6只二极管能接18键,9只二极管能接21键,第22键要单独占用3只二极管最不化算。

实验用89S51作试验,电路接线就是P1.2, P1.3, P1.4接键盘,P1.0接显示器。

/*==================================================================*

 *                   3个IO接识别22键测试程序                         *

 *       ------------------------------------------------            *

 *       MCU:     AT89C2051                                          *

 *       OSC:     12M cysytel                                        *

 *       程序设计:Cowboy                                             *

 *       程序版本:V1.0                                               *

 *==================================================================*/

 #include <reg52.h>

 //================== IO口线连接 ==================

sbit Bus          = P1^;

 sbit IO_a         = P1^;

 sbit IO_b         = P1^;

 sbit IO_c         = P1^;

 //================== 变量声明 ====================

unsigned char Disp_buf[];

 unsigned char Dig;

 unsigned char Key_count;

 unsigned char bdata Key_state;    

 sbit KB0 = Key_state^;

 sbit KB1 = Key_state^;

 sbit KB2 = Key_state^;

 sbit KB3 = Key_state^;

 sbit KB4 = Key_state^;

 sbit KB5 = Key_state^;

 //================== 表格数据 ====================

code unsigned char LED_font[]=

 {

         0x84,0x9f,0xa2,0x8a,0x99,0xc8,0xc0,0x9e,0x80, //

         0x88,0x90,0xc1,0xe4,0x83,0xe0,0xf0,0xff,0xfb, //9abcdef -

};

code unsigned char Key_tab[]=     //键码映射表

{//  0  1  2  3  4  5  6  7  8  9   

         , , , , , , , , , , //

          , , , , ,, , , , , //1X

          , , , , , , ,, , , //2X

         ,, , , , , , , ,, //3X

          , , , , ,, ,, , , //4X

          ,, , ,, , ,, , , //5X

         ,, ,                     //6X

 };

 //=============== 检测按键 =================

void Key_scan()

 {   

     unsigned char i;

     Key_count --;                        //扫描次序

    Key_count &= ;

     switch (Key_count)                //按次序处理

    {

         case :                                //第一轮扫描

        KB0 = IO_b; 

         KB1 = IO_c; 

         IO_a = ;

         IO_b = ;

         break;

         case :                                //每二轮扫描

        KB2 = IO_c;

         KB3 = IO_a;

         IO_b = ;

         IO_c = ;

         break;

         case :                                //每三轮扫描

        KB4 = IO_a;

         KB5 = IO_b;

         IO_c = ;

         break;

         default:                        //每四轮扫描

        if (!IO_a) KB0 = ;

         if (!IO_b) KB2 = ;

         if (!IO_c) KB4 = ;

         IO_a = ;

         //======更新显示缓冲区=======

         i = Key_tab[Key_state];

         if (i == )

         {

             Disp_buf[] = 0x11;                //显示三横

            Disp_buf[] = 0x11;

             Disp_buf[] = 0x11;

         }

         else

         {

             Disp_buf[] = 0x0c;     //字符"C"

             Disp_buf[] = i / ;   //键码十位

            Disp_buf[] = B;于      //键码个位

        }

         Key_state = ;

     }

 }    

 /*===================================================================

                     ONE WIRE 显示总线驱动程序       

 ===================================================================*/

 //=============== 发送一位 =================

void Send_bit(bit Dat)    

 {    

     unsigned char i = ;

     if (!Dat) Bus = ;

     else

     {

         Bus = ;

         Bus = ;

     }

     while(--i);                 //延时8us    

     Bus = ;

 }    

 //=============== 总线驱动 =================

void Bus_drive()

 {

     unsigned char i = ;

     unsigned char Sdat;

     Send_bit();                        //Bit6消隐

    do Bus = ; while(--i);             //延时768us

     do Bus = ; while(--i);             //延时768us

     Bus = ;

     Sdat = LED_font[Disp_buf[Dig++]];   //获取显示数据

    Send_bit(Sdat & 0x01);              //发送位0        

     Send_bit(Sdat & 0x02);              //发送位1        

     Send_bit(Sdat & 0x04);              //发送位2        

     Send_bit(Sdat & 0x08);              //发送位3        

     Send_bit(Sdat & 0x10);              //发送位4        

     Send_bit(Sdat & 0x20);              //发送位5        

     Send_bit(Dig  & 0x01);              //发送位选1        

     Send_bit(Dig  & 0x02);              //发送位选2

     while(--i);                         //延时512us

     Send_bit(Sdat & 0x40);              //发送位6

     for (i = ;i> ;i--) Send_bit();  //位6移至Dout

     if (Dig == ) Dig = ;

 }     

 /*===================================================================

                     延时 5ms 程序       

 ===================================================================*/

void Delay_5ms()        

 {    

     while(!TF1);    

     TF1 = ;    

     TH1 = (- ) / ;

     TL1 = (- ) % ;

 }    

 /*===================================================================

                         主程序       

 ===================================================================*/

void main()

 {

     TMOD = 0x10;            //定时器1,16位模式

    TCON = 0xc0;            //TR1=1;TF1=1;

     while()                //主循环

    {

         Bus_drive();        //显示总线驱动 

        Key_scan();         //检测按键

        Delay_5ms();        //延时5MS    

     }

 }

【29楼】我把22个按键的组态描述一下,看图就不会觉得费劲了

三个IO简称为A,B,C

按键1:A直接接地

按键2:A、B通过两二极管同时接地

按键3:B直接接地

按键4:B、C通过两二极管同时接地

按键5:C直接接地

按键6:C、A通过两二极管同时接地

按键7:B通过二极管被A拉低

按键8:A通过二极管被B拉低

按键9:C通过二极管被B拉低

按键10:B通过二极管被C拉低

按键11:A通过二极管被C拉低

按键12:C通过二极管被A拉低

按键13:A、B直接短路

按键14:B、C直接短路

按键15:C、A直接短路

按键16:B、C通过两二极管同时被A拉低

按键17:C、A通过两二极管同时被B拉低

按键18:A、B通过两二极管同时被C拉低
按键19:A通过二极管被B或C拉低

按键20:B通过二极管被C或A拉低

按键21:C通过二极管被A或B拉低

按键22:A、B、C通过三个二极管(或电阻)同时接地

两个二极管的2个IO的情况可参照以前的三菱键盘的帖子:

http://www.amobbs.com/bbs/bbs_content.jsp?bbs_sn=1280358

3个普通IO识别22个按键试验(转)的更多相关文章

  1. android蓝牙耳机下的语音(输入/识别)及按键监听

    背景:本人负责公司android平台的app开发,最近要开发一个语音助手类的app,类似于灵犀语音助手.虫洞语音助手等.其中有两个蓝牙耳机下的语音识别问题,比较折腾人,问题描述:1.蓝牙耳机连接下捕获 ...

  2. java io系列22之 FileReader和FileWriter

    FileReader 是用于读取字符流的类,它继承于InputStreamReader.要读取原始字节流,请考虑使用 FileInputStream.FileWriter 是用于写入字符流的类,它继承 ...

  3. 3个IO口8个按键

  4. stm32按键识别

    刚写了一个关于stm32单片机的按键识别的程序.目的,同时识别多个按键,并且不浪费cpu的时间. 关于去抖动,以前以为是在按键的时候,手会抖动.通过程序验证,这个确实是误解.这个应该是防止意外干扰.以 ...

  5. 【转】android 物理按键

    关键词:android   按键  矩阵按键 AD按键  平台信息: 内核:linux2.6/linux3.0 系统:android/android4.0 平台:S5PV310(samsung exy ...

  6. stm32寄存器版学习笔记01 GPIO口的配置(LED、按键)

    STM32的I/O口可以由软件配置成如下8种模式:输入浮空.输入上拉.输入下拉.模拟输入.开漏输出.推挽输出.推挽式复用功能及开漏复用功能.每个I/O口由7个寄存器来控制:配置模式的端口配置寄存器CR ...

  7. 让MacBook识别noppoo mini 84

    让MacBook识别noppoo mini 84 noppoo默认是没有mac驱动的,需要下载一个IOUSBHIDDriverDescriptorOverride否则无法noppoo的键位是识别错误的 ...

  8. 09A-独立按键消抖实验01——小梅哥FPGA设计思想与验证方法视频教程配套文档

    芯航线--普利斯队长精心奉献   实验目的: 1.复习状态机的设计思想并以此为基础实现按键消抖 2.单bit异步信号同步化以及边沿检测 3.在激励文件中学会使用随机数发生函数$random 4.仿真模 ...

  9. zigbee学习之路(三):按键的控制

    一.前言 通过前一次的实验,相信大家都已经对cc2530程序的编写有了一定的认识,这次我们来操作和实验的是cc2530上的按键模块. 二.原理分析 我们先来看一下按键的原理图: 根据原理图我们可以得出 ...

随机推荐

  1. ZOJ 1967 POJ 2570 Fiber Network

    枚举起点和公司,每次用DFS跑一遍图,预处理出所有的答案.询问的时候很快就能得到答案. #include<cstdio> #include<cmath> #include< ...

  2. MyBatis学习-SQL 符号篇

    当我们需要通过 XML 格式处理 SQL 语句时,经常会用到 <,<=,>,>= 等符号,但是很容易引起 XML 格式的错误,这样会导致后台将 XML 字符串转换为 XML文档 ...

  3. ios使用xcode进行Archive打包上传出现的常见错误

    error itms 90362上传appstore 一直报错ERROR ITMS-90362: "Invalid Info.plist value. The value for the k ...

  4. MFC中MessageBeep与sndPlaySound播放声音函数使用

    MessageBeep(0x00000000L);    //用来播放系统默认音频文件,如0x00000000L为系统提示音,具体音频对应规则,请参照MSDN. sndPlaySound函数用来播放指 ...

  5. JDK和JRE的区别

    JDK(Java Development Kit)是针对Java开发员的产品,是整个Java的核心,包括了Java运行环境JRE.Java工具和Java基础类库.Java Runtime Enviro ...

  6. 通过纯Java代码从excle中读取数据(为.xlsx文件)

    参考链接: 程序代码: package demo; import java.io.File; import java.io.IOException; import java.io.InputStrea ...

  7. CentOS中由一般用户切换为root用户

    --->http://www.centoscn.com/CentOS/help/2014/0624/3173.html 1.打开终端,提示符为“$”,表明该用户为普通用户,此时,直接输su,回车 ...

  8. TextBox只读时不能通过后台赋值取值解决办法

    给页面的TextBox设置ReadOnly="True"时,在后台代码中不能赋值取值,下边几种方法可以避免:  1.不设置ReadOnly,设置onfocus=this.blur( ...

  9. project文件问题

    到编译文件中看看,红色的删除,坤哥这样做的 .

  10. chrome浏览器调试功能之后端篇

    作为后端开发人员,可能有很多同学不怎么了解chrome调试功能,而即将成为大神的我们,怎么也得会,知其然更要知其所以然,今天我带领大家好好的梳理一下,chrome浏览器调试,个人把它分成了前端功能和后 ...