/*采用行扫描法RAM占用空间较少(呵呵起码我写的程序是这样的,因为我刚开始学,还不是很懂各方面的程序优化)*/
#include <pic.h>
__CONFIG(0x3F32); //芯片配置字
#define uchar unsigned char
#define uint unsigned int
void delay10ms(uchar x);
void beep();
void init();
uchar key;
uchar LED_CODE[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,
0x80,0x90,0xbf};
void keyscan()
{
uchar temp;
PORTB=0x07;//将列线置0,行线作为输入状态检测。
temp=PORTB;//读回B端口状态值
if((temp&0x07)!=0x07)//将读取的值跟0x07做与运算然后再判断是否不等于0x07,如果不等于说明有按键按下
{
delay10ms(1);//延时10毫秒,去抖动(去干扰)
PORTB=0x07;//同上
temp=PORTB;//同上
if((temp&0x07)!=0x07)//延时消抖后再次判断是否真的有按键按下,如果有再判断是此行的哪个按键(这里我们用的是3x3矩阵键盘,则每行有3种情况)
{
PORTB=0x37;//第一行状态
temp=PORTB;//读回B端口状态值
switch(temp)//判断第一行按键按下后可能产生的情况
{
case 0x33:key=7;break;//第一种情况:1号按键按下_0B0011 0011
case 0x35:key=4;break;//第二种情况:2号按键按下_0B0011 0101
case 0x36:key=1;break;//第三种情况:3号按键按下_0B0011 0110
}
PORTB=0x2F;//第二行状态
temp=PORTB;//读回B端口状态值
switch(temp)//判断第二行按键按下后可能产生的情况
{
case 0x2b:key=8;break;//第一种情况:4号按键按下_0B0010 1011
case 0x2d:key=5;break;//第二种情况:5号按键按下_0B0010 1101
case 0x2e:key=2;break;//第三种情况:6号按键按下_0B0011 1110
}
PORTB=0x1F;//第三行状态
temp=PORTB;//读回B端口状态值
switch(temp)//判断第三行按键按下后可能产生的情况
{
case 0x1b:key=9;break;//第一种情况:7号按键按下_0B0001 1011
case 0x1d:key=6;break;//第二种情况:8号按键按下_0B0001 1101
case 0x1e:key=3;break;//第三种情况:9号按键按下_0B0001 1110
}
}
}
}
void main()
{
init();//系统初始化
PORTD=LED_CODE[10];//上电数码管显示“—”
while(1)
{
PORTB=0x07;
if(PORTB!=0x07)//判断有无按键按下有则执行按键处理程序
{
keyscan();//调用扫描子程序
PORTD=LED_CODE[key];//显示对应键号
PORTB=0x07;//重新赋值
if(PORTB!=0x07)//再次判断有无按键按下 按下蜂鸣器响一声
{
beep(); //调用蜂鸣器子函数
}
while(PORTB!=0x07)//按键松手检测,如果按键一直按下关闭蜂鸣器
{
RE1=0;
delay10ms(1);
}
}
}
}
void init()//系统初始化程序
{
TRISB=0x07;//低三位,行作为输入
TRISD=0;//D端口设置为输出
TRISA=0;//A端口设置为输出
TRISE=0;//E端口设置为输出
PORTA=0;//A端口全部设置为低电平,位选开放
PORTD=0xFF;//D端口全部设置为高电平,段选关闭数码管不亮。
RE1=0;//蜂鸣器关闭
}
void delay10ms(uchar x)
{
uint j,h;
for(j=0;j<x;j++)
for(h=0;h<625;h++);
}
void beep()
{
RE1=1;
delay10ms(10);
RE1=0;
delay10ms(10);
}
