8個IO口檢測64個按鍵,數位管顯示(Proteus仿真)
- 采用AT89C51
- Proteus仿真
執行個體代碼
#include <reg52.h>
#include<intrins.h>
#define uchar unsigned char
#define uint unsigned int
uchar table[]= {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};//0-9共陰數位管
uchar duanZhi[]= {0,0};//儲存每段數位管顯示位數的數值
sbit P36=P3^6;//數位管時能端
sbit P37=P3^7;
sbit ST=P3^0;//定義74HC595移位寄存器
sbit SH=P3^2;
sbit DS=P3^1;
sbit P33=P3^3;
sbit P34=P3^4;
sbit P35=P3^5;
uchar shi,ge;//數位管個位和十位顯示
uchar tmp;//暫存P0的值
static uchar keynum=0;//按鍵值
//毫秒級延時
void delay(uint z)
{
uint x,y;
for(x=z; x>>0; x--)
for(y=110; y>>0; y--);
}
void SendTo595(uchar byteData);
/*----------------------------------------------------------------------------------
顯示
----------------------------------------------------------------------------------*/
void display()
{
ge = keynum%10;
shi = keynum/10;
duanZhi[0]=table[ge];
duanZhi[1]=table[shi];
//顯示個位
P37=0;
SendTo595(duanZhi[0]); //
delay(2);
P37=1;//消隐
//顯示十位
P36=0;
SendTo595(duanZhi[1]);//
delay(2);
P36=1;//消隐
}
/***********************************************************
*函數名 :SendTo595
*功能 :串行發送8個比特(一個位元組)的資料給595,再并行輸出
*參數 :byteData
************************************************************/
void SendTo595(uchar byteData)
{
uchar i=0;
ST = 0; //ST //先拉低,為後面的上升沿做準備
for(i; i<8; i++)
{
SH = 0;//先拉低,
if(byteData&0x80)DS=1;
else DS=0;
// DS = (byteData&0x80)?1:0;
byteData = byteData <<1; //該位元組右移一位
SH = 1;//上升沿,讓串行輸入時鐘變為高電平,并延時2個時鐘周期
_nop_();
_nop_();
SH = 0; //上升沿,讓串行輸入時鐘變為高電平,并延時2個時鐘周期
}
/*位移寄存器資料準備完畢,轉移到存儲寄存器*/
ST =1;
_nop_();
_nop_();
ST = 0;
}
void key_scan()
{
P0=0xff;
delay(6);
if(P0!=0xff)
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xfe:
keynum=57;
break;//第1行第1個按鍵按下
case 0xfd:
keynum=58;
break;//第1行第2個按鍵按下
case 0xfb:
keynum=59;
break;//第1行第3個按鍵按下
case 0xf7:
keynum=60;
break;//第1行第4個按鍵按下
case 0xef:
keynum=61;
break;//第1行第5個按鍵按下
case 0xdf:
keynum=62;
break;//第1行第6個按鍵按下
case 0xbf:
keynum=63;
break;//第1行第7個按鍵按下
case 0x7f:
keynum=64;
break;//第1行第8個按鍵按下
}
}
/********************第1行掃描**************************/
P0=0xef;//掃描第1行
delay(6);
if(P0!=0xef)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0x7e:
keynum=50;
break;//第1行第1個按鍵按下
case 0x7d:
keynum=51;
break;//第1行第2個按鍵按下
case 0x7b:
keynum=52;
break;//第1行第3個按鍵按下
case 0x77:
keynum=53;
break;//第1行第4個按鍵按下
case 0x6f:
keynum=54;
break;//第1行第5個按鍵按下
case 0x5f:
keynum=55;
break;//第1行第6個按鍵按下
case 0x3f:
keynum=56;
break;//第1行第7個按鍵按下
}
}
/********************第2行掃描**************************/
delay(6);
P0=0xbf;//掃描第2行
if(P0!=0xbf)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0x7e:
keynum=43;
break;//第2行第1個按鍵按下
case 0x7d:
keynum=44;
break;//第2行第2個按鍵按下
case 0x7b:
keynum=45;
break;//第2行第3個按鍵按下
case 0x77:
keynum=46;
break;//第2行第4個按鍵按下
case 0x6f:
keynum=47;
break;//第2行第5個按鍵按下
case 0x5f:
keynum=48;
break;//第2行第6個按鍵按下
case 0x3f:
keynum=49;
break;//第2行第7個按鍵按下
}
}
/********************第3行掃描**************************/
P0=0xdf;//掃描第3行
delay(6);
if(P0!=0xdf)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xde:
keynum=36;
break;//第3行第1個按鍵按下
case 0xdd:
keynum=37;
break;//第3行第2個按鍵按下
case 0xdb:
keynum=38;
break;//第3行第3個按鍵按下
case 0xd7:
keynum=39;
break;//第3行第4個按鍵按下
case 0xcf:
keynum=40;
break;//第3行第5個按鍵按下
case 0x9f:
keynum=41;
break;//第3行第6個按鍵按下
case 0x5f:
keynum=42;
break;//第3行第7個按鍵按下
}
}
/********************第4行掃描**************************/
P0=0xef;//掃描第4行
delay(6);
if(P0!=0xef)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xee:
keynum=29;
break;//第4行第1個按鍵按下
case 0xed:
keynum=30;
break;//第4行第2個按鍵按下
case 0xeb:
keynum=31;
break;//第4行第3個按鍵按下
case 0xe7:
keynum=32;
break;//第3行第4個按鍵按下
case 0xcf:
keynum=33;
break;//第4行第5個按鍵按下
case 0xaf:
keynum=34;
break;//第4行第6個按鍵按下
case 0x6f:
keynum=35;
break;//第4行第7個按鍵按下
}
}
/********************第5行掃描**************************/
P0=0xf7;//掃描第4行
delay(6);
if(P0!=0xf7)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xf6:
keynum=22;
break;//第5行第1個按鍵按下
case 0xf5:
keynum=23;
break;//第5行第2個按鍵按下
case 0xf3:
keynum=24;
break;//第5行第3個按鍵按下
case 0xe7:
keynum=25;
break;//第5行第4個按鍵按下
case 0xd7:
keynum=26;
break;//第5行第5個按鍵按下
case 0xb7:
keynum=27;
break;//第5行第6個按鍵按下
case 0x77:
keynum=28;
break;//第5行第7個按鍵按下
}
}
/********************第6行掃描**************************/
P0=0xfb;//掃描第4行
delay(6);
if(P0!=0xfb)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xfa:
keynum=15;
break;//第6行第1個按鍵按下
case 0xf9:
keynum=16;
break;//第6行第2個按鍵按下
case 0xf3:
keynum=17;
break;//第6行第3個按鍵按下
case 0xeb:
keynum=18;
break;//第6行第4個按鍵按下
case 0xdb:
keynum=19;
break;//第6行第5個按鍵按下
case 0xbb:
keynum=20;
break;//第6行第6個按鍵按下
case 0x7b:
keynum=21;
break;//第6行第7個按鍵按下
}
display();//這裡的顯示是儲存上一次的顯示值
}
/********************第7行掃描**************************/
P0=0xfb;//掃描第7行
delay(6);
if(P0!=0xfb)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xfc:
keynum=8;
break;//第7行第1個按鍵按下
case 0xf9:
keynum=9;
break;//第7行第2個按鍵按下
case 0xf5:
keynum=10;
break;//第7行第3個按鍵按下
case 0xed:
keynum=11;
break;//第7行第4個按鍵按下
case 0xdd:
keynum=12;
break;//第7行第5個按鍵按下
case 0xbd:
keynum=13;
break;//第7行第6個按鍵按下
case 0x7d:
keynum=14;
break;//第7行第7個按鍵按下
}
}
/********************第8行掃描**************************/
P0=0xfe;//掃描第8行
delay(6);
if(P0!=0xfe)//有按鍵按下
{
tmp=P0;//将檢測到的P0狀态值指派給臨時變量
switch(tmp)
{ //臨時變量對逐個IO口進行查詢
case 0xfc:
keynum=1;
break;//第8行第1個按鍵按下
case 0xfa:
keynum=2;
break;//第8行第2個按鍵按下
case 0xf6:
keynum=3;
break;//第8行第3個按鍵按下
case 0xee:
keynum=4;
break;//第8行第4個按鍵按下
case 0xde:
keynum=5;
break;//第8行第5個按鍵按下
case 0xbe:
keynum=6;
break;//第8行第6個按鍵按下
case 0x7e:
keynum=7;
break;//第8行第7個按鍵按下
}
}
display();//這裡的顯示是儲存上一次的顯示值
}
void main()
{
keynum=0;
P0=0xff;
while(1)
{
key_scan();
}
}
proteus仿真軟體裡面有些bug
- 注意:二極管選型,不能選擇發光二極管,有些其他的二極管也不好使,我使用的是
10A01
- 注意走線,線路連接配接中最好不要出現重複線或者點,有多餘的線或者連接配接點最好删除,不然出問題了排查問題帶來難度,有可能也會影響仿真效果,電阻選擇不能過大。仿真時,會比較占電腦記憶體。按鍵操作和響應速度可能存在不同步或跟不上,