单片机系统包括输出与输入系统,在学习过的LED、LCD显示等都为输出系统,常见的输入系统有键盘等。基于以前学习的内容,加上键盘输入系统构成简单的电子密码锁。在键盘输入设计中要注意去抖动和判断是否一次按键而有多个键值输入的情况。
主要器件:
1、AT89C52单片机芯片,接受键盘的输入并作出判断,并控制6位7段数码管和LED指示灯的显示。
2、4×4行列式键盘。
3、7段数码管,共6位数码管,对应于6位密码。
4、LED指示灯,红灯表示输入密码错误告警,绿灯表示输入密码正确。
5、CMOS六反相驱动器CD4069UB,以提供足够的驱动电流。
试验流程图:
试验电路图:
试验程序代码:
//PWLock.h程序
#ifndef _PWLOCK_H // 防止PWLock.h被重复引用
#define _PWLOCK_H
#include
#include
#include
#include
#define uchar unsigned char
#define uint unsigned int
uchar digbit; // 字位
uchar wordbuf[6]; // 字型码缓冲区
uchar t1count; // 定时器1由50ms累积到1s所用的计数器
uchar count; // 密码位计数
uchar pw[6]; // 初始密码存储区
uchar pwbuf[6]; // 输入密码存储区
bit enterflag; // 确认键按下与否标志
bit pwflag; // 密码正确与否标志
bit showflag; // 数码管显示与否标志
sbit green=P3^0;
sbit red=P3^1;
void display(void); // 显示函数
#endif
//PWLock.c程序
#include "PWLock.h"
/* 键消抖延时函数 */
void delay(void)
{ uchar i;
for (i=300;i>0;i--);
}
/* 键扫描函数 */
uchar keyscan(void)
{ uchar keyl,key0;
uchar keyr;
P1=0x0f;
if ((P1|0x0f)!=0x0f) // 若有键按下................
{
{
delay(); //去抖
if((P1|0x0f)!=0x0f)
keyr=P1&0xf0;
}
P1=0xf0;
if((P1|0xf0)!=0xf0)
{
delay(); //去抖动
if((P1|0xf0)!=0xf0)
keyl=P1&0x0f;
}
}
key0=keyr+keyl;
return key0;
}
/* 定时器0中断服务子程序,2ms定时动态扫描显示 */
void time0_int(void) interrupt 1
{ /* 重置2ms定时 */
TH0 = -2000/256;
TL0 = -2000%256;
if (showflag==1)
display(); // 调用显示函数
else
digbit=0x01;
}
/* 定时器1中断服务子程序,50ms*/
void time1_int(void) interrupt 3
{ uchar k;
/* 重置50ms定时 */
TH1 = -50000/256;
TL1 = -50000%256;
if (t1count<100)
{ t1count++;
}
else // 计时到5s
{ TR1 = 0; // 关闭计数器1
t1count = 0;
green = 1; // 绿灯不亮
red = 1; // 红灯不亮
showflag = 1; // 打开数码管显示
digbit = 0x01; // 从数码管第1位开始动态显示
for (k=0;k<6;k++) // 显示888888
wordbuf[k]=8;
}
}
/* 根据共阴极字型编码表获取0~9,A~B字型代码 */
uchar getcode(uchar i)
{ uchar p;
switch (i)
{ case 0: p=0xc0; break; /* 0 */
case 1: p=0xf9; break; /* 1 */
case 2: p=0xa6; break; /* 2 */
case 3: p=0xb0; break; /* 3 */
case 4: p=0x99; break; /* 4 */
case 5: p=0x92; break; /* 5 */
case 6: p=0x82; break; /* 6 */
case 7: p=0xf8; break; /* 7 */
case 8: p=0x80; break; /* 8 */
case 9: p=0x90; break; /* 9 */
case 10: p=0x88; break; /* A */
case 11: p=0x83; break; /* B */
case 12: p=0xc6; break; /* C */
case 13: p=0xa1; break; /* D */
case 14: p=0x86; break; /* E */
case 15: p=0x8e; break; /* F */
default: break;
}
return(p);
}
/* 显示函数 */
void display(void)
{ uchar i;
switch (digbit)
{ case 1: i=0; break;
case 2: i=1; break;
case 4: i=2; break;
case 8: i=3; break;
case 16: i=4; break;
case 32: i=5; break;
default: break;
}
P2 = 0x00; // 关闭显示
P0 = getcode(wordbuf[i]); // 送字型码
P2 = digbit; // 送字位码
if (digbit<0x20) // 共6位
digbit = digbit*2; // 左移一位
else
digbit = 0x01;
}
/* 密码比较函数 */
bit pwcmp(void)
{ bit flag;
uchar i;
for (i=0;i<6;i++)
{
if (pw[i]==pwbuf[i])
flag = 1;
else
{
flag = 0;
i = 6;
}
}
return(flag);
}
接上篇程序代码:
/* 主程序 */
void main()
{
uchar j,key;
P2 = 0x00; // 关闭数码管显示
TMOD = 0x11; // T0,T1工作方式1
/* 2ms 定时设置 */
TH0 = -2000/256;
TL0 = 2000%256;
/* 50ms 定时设置 */
TH1 = -50000/256;
TL1 = -50000%256;
/* 启动计数器0,关闭计数器1 */
TR0 = 1;
ET0 = 1;
TR1 = 0;
ET1 = 1;
EA = 1;
count = 0; // 初始没有输入密码,计数器设为0
enterflag = 0; // 没有按下确认键
pwflag = 0; // 密码标志先置为0
green = 1; // 绿灯不亮
red = 1; // 红灯不亮
/* 假设内定密码为164211 */
pw[0] = 1;
pw[1] = 6;
pw[2] = 4;
pw[3] = 2;
pw[4] = 1;
pw[5] = 1;
digbit = 0x01; // 从第一位数码管开始动态扫描
/* 刚加电时,显示888888 */
for (j=0;j<6;j++)
wordbuf[j] = 8;
showflag = 1; // 打开数码管显示
for(;;)
{
key=keyscan(); // 调用键盘扫描函数
if(P1==0xf0) //确定按键已经放开
switch(key)
{
case 0x11: // 1行1列,数字0
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示?
pwbuf[count] = 0;
count++;
break;
}
case 0x21: // 1行2列,数字1
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F
pwbuf[count] = 1;
count++;
}
break;
case 0x41: // 1行3列,数字2
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 2;
count++;
}
break;
case 0x81: // 1行4列,数字3
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 3;
count++;
}
break;
case 0x12: // 2行1列,数字4
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 4;
count++;
}
break;
case 0x22: // 2行2列,数字5
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 5;
count++;
}
break;
case 0x42: // 2行3列,数字6
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 6;
count++;
}
break;
case 0x82: // 2行4列,数字7
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 7;
count++;
}
break;
case 0x14: // 3行1列,数字8
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 8;
count++;
}
break;
case 0x24: // 3行2列,数字9
if (count<6)
{
wordbuf[count] = 0x0f; // 对应密码位上显示“F"
pwbuf[count] = 9;
count++;
}
break;
case 0x44: // 3行3列,确认键
enterflag = 1; // 确认键按下
if (count==6) // 只有输入6个密码后按确认键才作密码比较
pwflag = pwcmp();
else
pwflag = 0; // 否则直接pmflag赋0
break;
case 0x84: // 3行4列,取消键
count = 0; // 密码计数清零
for (j=0;j<6;j++)
{
wordbuf[j] = 8; // 数码管显示888888
pwbuf[j] = 0x0f; // 用FFFFFF清除已经输入的密码
}
break;
default:break;
}
if (enterflag==1) // 如果按下确认键
{
enterflag = 0; // 标志位置回0
count = 0; // 密码位计数器清零
for (j=0;j<6;j++)
pwbuf[j] = 0x0f; // 用FFFFFF清除已经输入的密码
// showflag=0; // 关闭数码管显示
TR1=1; // 计数器1开始计数
t1count=0; // 定时器1由50ms累积到5s所用的计数器
if (pwflag==1)
green=0; // 绿灯亮
else
red = 0; // 红灯亮
}
}
}