LQB,手打,矩阵键盘,要P30,要P31。按键,数码管显示按键键码

蓝桥杯的矩阵键盘,一般不考长按的那些操作。只是几个按键的调换
所以这里用一种比较方便的方法。

也是延时去抖动的方法吧。
采用逐行扫描的方法;
先让有个IO口等于0,判别,列是不是有一个等于0,是的话,返回对应的按键键码;
最后,如果一个都没有按下来,那么返回0

#include <stc15.h>
#include <intrins.h>
//接口
sbit S7=P3^0;
sbit S6=P3^1;
sbit S5=P3^2;
sbit S4=P3^3;
//宏定义
#define     LED       4   //定义主时钟
#define     ULN       5   //定义主时钟
#define     COM       6   //定义主时钟
#define     ABC       7   //定义主时钟
typedef     unsigned char   u8;
typedef     unsigned int    u16;
typedef     unsigned long   u32;
//-----------------------------------------------

#define FOSC 11059200L

#define T1MS (65536-FOSC/1000)      //1T模式
//#define T1MS (65536-FOSC/12/1000) //12T模式

//全局变量
u8 LEDbuf=0xFF;
u8 ULNbuf=0x00;
u8 SMGbuf[20];
u8 SMGdat[8];//8个数码管,所以8个元素

u32 tim1ms=0;//u32类型,几十天才溢出

u8 keyval=0;
/*************  本地常量声明   ,这是共阴的 **************/
u8 code table[]={                       //标准字库
//   0    1    2    3    4    5    6    7    8    9    A    B    C    D    E    F
    0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71,
//black  -     H    J    K    L    N    o   P    U     t    G    Q    r   M    y
    0x00,0x40,0x76,0x1E,0x70,0x38,0x37,0x5C,0x73,0x3E,0x78,0x3d,0x67,0x50,0x37,0x6e,
    0xBF,0x86,0xDB,0xCF,0xE6,0xED,0xFD,0x87,0xFF,0xEF,0x46};    //0. 1. 2. 3. 4. 5. 6. 7. 8. 9. -1

u8 code T_COM[]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};      //位码
//函数声明
void delayms(u16 tms);
void Delay1ms();		//@11.0592MHz
void sel(u8 x); //选择四个分支哪一个?
void sysinit();//系统初始化,关闭所有的外部设备
void disp();
u8 keyScan();//矩阵键盘
//主函数
void main()
{
	//main局部变量
	
	unsigned int wendu=1025;
	//外设初始化
	  sysinit();
	
	//系统定时器初始化
	  AUXR |= 0x40;                   //定时器1为1T模式
    TMOD = 0x00;                    //设置定时器为模式0(16位自动重装载)
    TL1 = T1MS;                     //初始化计时值
    TH1 = T1MS >> 8;
    TR1 = 1;                        //定时器1开始计时
    ET1 = 1;                        //使能定时器0中断
    EA = 1;
	//大循环
	//这里,可以加一点测试代码,测试是不是每个外设都成功;
  	

	while(1)
	{
	
	 keyval= keyScan();
	  SMGdat[0]=0x00;
				SMGdat[1]=0x00 ;//添加小数点
			
				SMGdat[2]=table[keyval/10%10];
			
				SMGdat[3]= table[keyval%10];
				SMGdat[4]= 0x39;  //C
				SMGdat[5]= 0x00;
			
				SMGdat[6]= 0x00;
				SMGdat[7]= 0x00;
	  if(tim1ms%500==0)  //500ms读取一次数据,刷新数据
		{
		  
      

		}
	 
	
	 	  

	}
	
	
	
}

//函数定义
//-----------------------------------------------

/* Timer1 interrupt routine */
void tm1_isr() interrupt 3 using 1
{
    
	  tim1ms++;
	  disp();//1ms显示一次数码管
}
void Delay1ms()		//@11.0592MHz
{
	unsigned char i, j;

	_nop_();
	_nop_();
	_nop_();
	i = 11;
	j = 190;
	do
	{
		while (--j);
	} while (--i);
}

void delayms(u16 tms)
{
	u16 i=0;
	for(i=0;i<tms;i++)
	{
		  Delay1ms();
	}
}

void sel(u8 x)  //选择四个分支哪一个?
{
	//背下来了吗?
	switch(x)
	{
		case 0:P2=P2&0x1F;break;
		case LED:P2=P2&0x1F|0x80;break;
		case ULN:P2=P2&0x1F|0xA0;break;
		case COM:P2=P2&0x1F|0xC0;break;
		case ABC:P2=P2&0x1F|0xE0;break;
	}
}
void sysinit()//系统初始化,关闭所有的外部设备
{
	sel(LED);LEDbuf=0xFF;P0=LEDbuf;sel(0);
  sel(ULN);ULNbuf=0x00;P0=ULNbuf;sel(0);
	sel(COM);P0=0x00;sel(0);
	sel(ABC);P0=0xFF;sel(0);
}


void disp()
{
	//背下来
	static u8 i=0;
	//消隐
	sel(COM);P0=0x00;sel(0);
	sel(ABC);P0=0xFF;sel(0);
//显示
	sel(COM);P0=T_COM[i];sel(0);
	sel(ABC);P0=~SMGdat[i];sel(0);
	
	i++;
	if(i>=8)i=0;
}


u8 keyScan()
{
	P30=0;P31=1;P32=1;P33=1;
	P34=1;P35=1;P42=1;P44=1;
	if(P34==0)
	{
	   delayms(10);
		 if(P34==0)
		 return 19;
	}
	else 	if(P35==0)
	{
	   delayms(10);
		 if(P35==0)
		 return 15;
	}
		else 	if(P42==0)
	{
	   delayms(10);
		 if(P42==0)
		 return 11;
	}
		else 	if(P44==0)
	{
	   delayms(10);
		 if(P44==0)
		 return 7;
	}
	
	
	
	P30=1;P31=0;P32=1;P33=1;
	P34=1;P35=1;P42=1;P44=1;
	if(P34==0)
	{
	   delayms(10);
		 if(P34==0)
		 return 18;
	}
	else 	if(P35==0)
	{
	   delayms(10);
		 if(P35==0)
		 return 14;
	}
		else 	if(P42==0)
	{
	   delayms(10);
		 if(P42==0)
		 return 10;
	}
		else 	if(P44==0)
	{
	   delayms(10);
		 if(P44==0)
		 return 6;
	}
	
	P30=1;P31=1;P32=0;P33=1;
	P34=1;P35=1;P42=1;P44=1;
	if(P34==0)
	{
	   delayms(10);
		 if(P34==0)
		 return 17;
	}
	else 	if(P35==0)
	{
	   delayms(10);
		 if(P35==0)
		 return 13;
	}
		else 	if(P42==0)
	{
	   delayms(10);
		 if(P42==0)
		 return 9;
	}
		else 	if(P44==0)
	{
	   delayms(10);
		 if(P44==0)
		 return 5;
	}
	
	P30=1;P31=1;P32=1;P33=0;
	P34=1;P35=1;P42=1;P44=1;
	if(P34==0)
	{
	   delayms(10);
		 if(P34==0)
		 return 16;
	}
	else 	if(P35==0)
	{
	   delayms(10);
		 if(P35==0)
		 return 12;
	}
		else 	if(P42==0)
	{
	   delayms(10);
		 if(P42==0)
		 return 8;
	}
		else 	if(P44==0)
	{
	   delayms(10);
		 if(P44==0)
		 return 4;
	}
	
	else return 0;  //没有按键按下,返回0
	
}


 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值