/*采用行扫描法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); }