#include<reg52.h>
#define uch unsigned char
#define uint unsigned int
#define ulong unsigned long
sbit ADDR0=P1^0;
sbit ADDR1=P1^1;
sbit ADDR2=P1^2;
sbit ADDR3=P1^3;
sbit ENLED=P1^4;
sbit KEY_IN_1=P2^4;
sbit KEY_IN_2=P2^5;
sbit KEY_IN_3=P2^6;
sbit KEY_IN_4=P2^7;
sbit key_out_1=P2^3;
sbit key_out_2=P2^2;
sbit key_out_3=P2^1;
sbit key_out_4=P2^0;
uch code ledchar[]={ //从数字0到F的显示码。
0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e
};
uch keysta[4][4]={
{1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1},
};
void main()
{
uch i,j;//循环变量i和j
uch backup[4][4]={
{1,1,1,1},{1,1,1,1},{1,1,1,1},{1,1,1,1},
};//按键值备份。
EA=1;//使能总中断
ENLED=0;//选择数码管DS1进行显示。
ADDR3=1;
ADDR2=0;
ADDR1=0;
ADDR0=0;
TMOD=0x01;//定时器0工作在模式1
TH0=0xfc;
TL0=0x67;//定时中断间隔1毫秒
ET0=1;//开定时器0中断。
TR0=1;//启动定时器0
P0=ledchar[0]; //LED显示0
while(1)
{
for (i=0;i<4;i++)//外层循环,执行四次;
{
for (j=0;j<4;j++)//内层循环,执行四次
{
if (backup[i]!=keysta[i][j])
{
if (backup[i][j]!=0)//如果上次保存的状态是非0,就是1,表明上次按钮按下,现在状态不同上次保存状态,那意思就是按下后已弹起
{
P0=ledchar[i*4+j];//将编号显示出来。i是行号,从0行开始,j是行中的第几个元素,那么变成一维数据就是行号*列宽+列号
} //判断按下弹起大括号
backup[i][j]=keysta[i][j];
}//判断键值变化大括号
} //内层循环大括呈
}//外层循环大括号
} //while大括号
} //main函数大括号
void InterruptTimer0() interrupt 1
{
uch i;
static uch keyout=0;
static uch keybuf[4][4]={
{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff},{0xff,0xff,0xff,0xff}
};
TH0=0xfc;//重新加载初值
TL0=0x67;
keybuf[keyout][0]=(keybuf[keyout][0]<<1)|KEY_IN_1;
keybuf[keyout][1]=(keybuf[keyout][0]<<1)|KEY_IN_2;
keybuf[keyout][2]=(keybuf[keyout][0]<<1)|KEY_IN_3;
keybuf[keyout][3]=(keybuf[keyout][0]<<1)|KEY_IN_4;
//消抖后更新按键状态
for (i=0;i<4;i++)//每行4个按键,所以循环4次
{
if ((keybuf[keyout][i]&0x0f)==0x00)
{ //连续4次扫描值为0,即4*4ms内都是按下状态时,可认为按键已稳定的弹起
keysta[keyout][i]=1;
}
}//外层循环大括号
//执行下一次的扫描输出
keyout++;
keyout=keyout&0x03;//索引值加到4即归零
switch(keyout)
{
case 0:key_out_4=1;key_out_1=0;break; //key_out_1低电平的时候,开始探测第一行按键。顺便把上次低电平的拉高。
case 1:key_out_1=1;key_out_2=0;break; //
case 2:key_out_2=1;key_out_3=0;break;
case 3:key_out_3=1;key_out_4=0;break;
default:break;
}//switch后大括号
}//中断大括号 |