【Proteus仿真】51单片机电子锁综合设计案例

【Proteus仿真】51单片机电子锁综合设计案例


  • 📽🎞📺🎬Proteus仿真演示
    在这里插入图片描述

📓功能介绍

🎉本示例包含LCD1602驱动显示,4x4矩阵按键扫描功能,AT24C02存储,外设蜂鸣器等驱动。

  • 📖程序框架
    在这里插入图片描述

⛳主程序代码

具体详情请参考源程序代码。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
//包含头文件
#include "i2c.h"
#include "LCD1602.h"

sbit ALAM = P2^1; //报警
sbit KEY = P3^6; //开锁

bit pass=0; //密码正确标志
bit ReInputEn=0; //重置输入允许标志
bit s3_keydown=0; //3秒按键标志位
bit key_disable=0; //锁定键盘标志

unsigned char countt0,second; //t0中断计数器,秒计数器

void Delay5Ms(void); //声明延时函数

unsigned char code a[]={0xFE,0xFD,0xFB,0xF7}; //控盘扫描控制表
//液晶显示数据数组
unsigned char code start_line[] = {"password: "};
unsigned char code name[] = {"===Elect Lock==="}; //显示名称
unsigned char code Correct[] = {" correct "}; //输入正确
unsigned char code Error[] = {" error "}; //输入错误
unsigned char code codepass[] = {" pass "};
unsigned char code LockOpen[] = {" open "}; //OPEN
unsigned char code SetNew[] = {"SetNewWordEnable"};
unsigned char code Input[] = {"input: "}; //INPUT
unsigned char code ResetOK[] = {"ResetPasswordOK "};
unsigned char code initword[] = {"Init password..."};
unsigned char code Er_try[] = {"error,try again!"};
unsigned char code again[] = {"input again "};

unsigned char InputData[6]; //输入密码暂存区
unsigned char CurrentPassword[6]={0,0,0,0,0,0}; //读取EEPROM密码暂存数组
unsigned char TempPassword[6];
unsigned char N=0; //密码输入位数记数
unsigned char ErrorCont; //错误次数计数
unsigned char CorrectCont; //正确输入计数
unsigned char ReInputCont; //重新输入计数
unsigned char code initpassword[6]={0,0,0,0,0,0}; //输入管理员密码后将密码初始为000000
unsigned char code adminpassword[6]={1,2,3,4,5,6}; //输入管理员密码后将密码初始为000000


//=====================5ms延时==============================
void Delay5Ms(void)
{
unsigned int TempCyc = 5552;
while(TempCyc--);
}

//===================400ms延时==============================
void Delay400Ms(void)
{
unsigned char TempCycA = 5;
unsigned int TempCycB;
while(TempCycA--)
{
TempCycB=7269;
while(TempCycB--);
}
}

//==============将按键值编码为数值=========================
unsigned char coding(unsigned char m)
{
unsigned char k;
switch(m)
{
case (0x11): k=1;break;//?0001 0001?
case (0x21): k=2;break;//?0010 0001?
case (0x41): k=3;break;//?0100 0001?
case (0x81): k=4;break;//?1000 0001?

case (0x12): k=5;break;
case (0x22): k=6;break;
case (0x42): k=7;break;
case (0x82): k=8;break;

case (0x14): k=9;break;//?0001 0100?
case (0x44): k='A';break; //?0010 0100?
case (0x24): k='B';break; //?0100 0100?
case (0x84): k='C';break;//?1000 0100?

case (0x18): k='\*';break;//?0001 1000?
case (0x28): k='D';break;//?0010 1000?
case (0x48): k='#';break;//?0100 1000?
case (0x88): k= 0;break;//?1000 1000?
}
return(k);
}
//=====================按键检测并返回按键值===============================
unsigned char keynum(void)
{
unsigned char row,col,i;
P1=0xf0;
if((P1&0xf0)!=0xf0)
{
Delay5Ms();
Delay5Ms();
if((P1&0xf0)!=0xf0)
{
row=P1^0xf0; //确定行线
i=0;
P1=a[i]; //精确定位
while(i<4)
{
if((P1&0xf0)!=0xf0)
{
col=~(P1&0xff); //确定列线
break; //已定位后提前退出
}
else
{
i++;
P1=a[i];
}
}
}
else
{
return 0;
}
while((P1&0xf0)!=0xf0);
return (row|col); //行线与列线组合后返回
}
else return 0; //无键按下时返回0
}
//=======================一声提示音,表示有效输入========================
void OneAlam(void)
{
ALAM=0;
Delay5Ms();
ALAM=1;
}
//========================二声提示音,表示操作成功========================
void TwoAlam(void)
{
ALAM=0;
Delay5Ms();
ALAM=1;
Delay5Ms();
ALAM=0;
Delay5Ms();
ALAM=1;
}
//========================三声提示音,表示错误========================
void ThreeAlam(void)
{
ALAM=0;
Delay5Ms();
ALAM=1;
Delay5Ms();
ALAM=0;
Delay5Ms();
ALAM=1;
Delay5Ms();
ALAM=0;
Delay5Ms();
ALAM=1;
}
//=======================显示提示输入=========================
void DisplayChar(void)
{
unsigned char i;
if(pass==1)
{
//DisplayListChar(0,1,LockOpen);
write\_1602com(er); //在二行开始显示
for(i=0;i<16;i++)
{
write\_1602dat(LockOpen[i]); //显示open 开锁成功

}
}
else
{
if(N==0)
{
//DisplayListChar(0,1,Error);
write\_1602com(er);
for(i=0;i<16;i++)
{
write\_1602dat(Error[i]); //显示错误
}
}
else
{
//DisplayListChar(0,1,start\_line);
write\_1602com(er);
for(i=0;i<16;i++)
{
write\_1602dat(start_line[i]);//显示开始输入
}
}
}
}

//========================重置密码==================================================
//==================================================================================
void ResetPassword(void)
{
unsigned char i;
unsigned char j;
if(pass==0)
{
pass=0;
DisplayChar(); //显示错误
ThreeAlam(); //没开锁时按下重置密码报警3声
}
else //开锁状态下才能进行密码重置程序
{
if(ReInputEn==1) //开锁状态下,ReInputEn置1,重置密码允许
{
if(N==6) //输入6位密码
{
ReInputCont++; //密码次数计数
if(ReInputCont==2) //输入两次密码
{
for(i=0;i<6;)
{
if(TempPassword[i]==InputData[i]) //将两次输入的新密码作对比
i++;
else //如果两次的密码不同
{
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(Error[j]); //显示错误Error
}
ThreeAlam(); //错误提示
pass=0; //关锁
ReInputEn=0; //关闭重置功能,
ReInputCont=0;
DisplayChar();
break;
}
}
if(i==6)
{
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(ResetOK[j]); //密码修改成功,显示
}

TwoAlam(); //操作成功提示
WrToROM(TempPassword,0,6); //将新密码写入24C02存储
ReInputEn=0;
}
ReInputCont=0;
CorrectCont=0;
}

else //输入一次密码时
{
OneAlam();
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(again[j]); //显示再输入一次
}
for(i=0;i<6;i++)
{
TempPassword[i]=InputData[i]; //将第一次输入的数据暂存起来
}
}

N=0; //输入数据位数计数器清零
}
}
}
}
//=======================输入密码错误超过三过,报警并锁死键盘======================
void Alam\_KeyUnable(void)
{
P1=0x00;
{
ALAM=~ALAM; //蜂鸣器一直闪烁鸣响
Delay5Ms();
}
}
//=======================取消所有操作============================================
void Cancel(void)
{
unsigned char i;
unsigned char j;
//DisplayListChar(0, 1, start\_line);
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(start_line[j]); //显示开机输入密码界面
}
TwoAlam(); //提示音
for(i=0;i<6;i++)
{
InputData[i]=0; //将输入密码清零
}
KEY=1; //关闭锁
ALAM=1; //报警关
pass=0; //密码正确标志清零
ReInputEn=0; //重置输入充许标志清零
ErrorCont=0; //密码错误输入次数清零
CorrectCont=0; //密码正确输入次数清零
ReInputCont=0; //重置密码输入次数清零
s3_keydown=0;
key_disable=0; //锁定键盘标志清零
N=0; //输入位数计数器清零
}

//==========================确认键,并通过相应标志位执行相应功能===============================
void Ensure(void)
{
unsigned char i,j;
RdFromROM(CurrentPassword,0,6); //从24C02里读出存储密码
if(N==6)
{
if(ReInputEn==0) //重置密码功能未开启
{
for(i=0;i<6;)
{
if(CurrentPassword[i]==InputData[i]) //判断输入密码和24c02中的密码是否相同
{
i++; //相同一位 i就+1
}
else //如果有密码不同
{
ErrorCont++; //错误次数++
if(ErrorCont==3) //错误输入计数达三次时,报警并锁定键盘
{
write\_1602com(er);
for(i=0;i<16;i++)
{
write\_1602dat(Error[i]);
}
do
Alam\_KeyUnable();
while(1);
}
else //错误次数小于3次时,锁死键盘3秒,然后重新可以输入
{
TR0=1; //开启定时
key_disable=1; //锁定键盘
pass=0; //pass位清零
break; //跳出
}
}
}

if(i==6) //密码输入正确时
{
if((InputData[0]==adminpassword[0])&&(InputData[1]==adminpassword[1])&&(InputData[2]==adminpassword[2])&&(InputData[3]==adminpassword[3])&&(InputData[4]==adminpassword[4])&&(InputData[5]==adminpassword[5]))
{
WrToROM(initpassword,0,6); //强制将初始密码写入24C02存储
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(initword[j]); //显示初始化密码
}
TwoAlam(); //成功提示音
Delay400Ms(); //延时400ms
TwoAlam(); //成功提示音
N=0; //输入位数计数器清零
}
else
{
CorrectCont++; //输入正确变量++
if(CorrectCont==1) //正确输入计数,当只有一次正确输入时,开锁
{
//DisplayListChar(0,1,LockOpen);
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(LockOpen[j]); //显示open开锁画面
}
TwoAlam(); //操作成功提示音
KEY=0; //开锁
pass=1; //置正确标志位
TR0=1; //开启定时
for(j=0;j<6;j++) //将输入清除
{
InputData[i]=0; //开锁后将输入位清零
}
}
else //当两次正确输入时,开启重置密码功能
{
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(SetNew[j]); //显示重置密码界面
}
TwoAlam(); //操作成功提示
ReInputEn=1; //允许重置密码输入
CorrectCont=0; //正确计数器清零
}
}
}
//=========================当第一次使用或忘记密码时可以用123456对其密码初始化============
else
{
if((InputData[0]==adminpassword[0])&&(InputData[1]==adminpassword[1])&&(InputData[2]==adminpassword[2])&&(InputData[3]==adminpassword[3])&&(InputData[4]==adminpassword[4])&&(InputData[5]==adminpassword[5]))
{
WrToROM(initpassword,0,6); //强制将初始密码写入24C02存储
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(initword[j]); //显示初始化密码
}
TwoAlam(); //成功提示音
Delay400Ms(); //延时400ms
TwoAlam(); //成功提示音
N=0; //输入位数计数器清零
}
else //密码输入错误
{
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(Error[j]); //显示错误信息
}
ThreeAlam(); //错误提示音
pass=0;
}
}
}

else //当已经开启重置密码功能时,而按下开锁键,
{
//DisplayListChar(0,1,Er\_try);
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(Er_try[j]); //错误,请重新输入
}
ThreeAlam(); //错误提示音
}
}

else //密码没有输入到6位时,按下确认键时
{
//DisplayListChar(0,1,Error);
write\_1602com(er);
for(j=0;j<16;j++)
{
write\_1602dat(Error[j]); //显示错误
}

ThreeAlam(); //错误提示音
pass=0;
}

N=0; //将输入数据计数器清零,为下一次输入作准备
}


//==============================主函数===============================
void main(void)
{
unsigned char KEY,NUM;
unsigned char i,j;
P1=0xFF; //P1口复位
TMOD=0x11; //定义工作方式
TL0=0xB0;
TH0=0x3C; //定时器赋初值
EA=1; //打开中断总开关
ET0=1; //打开中断允许开关
TR0=0; //打开定时器开关
Delay400Ms(); //启动等待,等LCM讲入工作状态
lcd\_init(); //LCD初始化
write\_1602com(yi);//日历显示固定符号从第一行第0个位置之后开始显示
for(i=0;i<16;i++)
{
write\_1602dat(name[i]);//向液晶屏写开机画面
}
write\_1602com(er);
for(i=0;i<16;i++)
{
write\_1602dat(start_line[i]);//写输入密码等待界面
}
write\_1602com(er+9); //设置光标位置
write\_1602com(0x0f); //设置光标为闪烁
Delay5Ms(); //延时片刻(可不要)

N=0; //初始化数据输入位数
while(1) //进入循环
{
if(key_disable==1) //锁定键盘标志为1时
Alam\_KeyUnable(); //报警键盘锁
else
ALAM=1; //关报警

KEY = keynum(); //读按键的位置码
if(KEY!=0) //当有按键按下时
{
if(key_disable==1) //锁定键盘标志为1时
{
second=0; //秒清零
}
else //没有锁定键盘时
{
NUM=coding(KEY); //根据按键的位置将其编码,编码值赋值给NUM
{
switch(NUM) //判断按键值
{
case ('A'): ; break;
case ('B'): ; break;
case ('C'): ResetPassword() ; break; //ABC是无定义按键
case ('D'): ResetPassword();break; //重新设置密码
case ('\*'): Cancel(); break; //取消当前输入
case ('#'): Ensure(); break; //确认键,
default: //如果不是功能键按下时,就是数字键按下
{
if(N<6) //当输入的密码少于6位时,接受输入并保存,大于6位时则无效。
{
write\_1602com(er);
for(i=0;i<16;i++)
{
write\_1602dat(Input[i]); //显示输入画面
}
OneAlam(); //按键提示音
for(j=0;j<=N;j++)
{
write\_1602com(er+6+j); //显示位数随输入增加而增加
write\_1602dat('\*'); //但不显示实际数字,用\*代替
// write\_1602dat(NUM +0x30); //显示按键输入的数字
}
InputData[N]=NUM; //将数字键的码赋值给InputData[]数组暂存
N++; //密码位数加
}
else //输入数据位数大于6后,忽略输入
{
N=6; //密码输入大于6位时,不接受输入
break;
}
}
}
}
}
}
}
}

//\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*中断服务函数\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*\*
void time0\_int(void) interrupt 1 //定时器T0
{
TL0=0xB0;
TH0=0x3C; //定时器重新赋初值
//TR0=1;
countt0++; //计时变量加,加1次时50ms
if(countt0==20) //加到20次就是1s
{
countt0=0; //变量清零
second++; //秒加
if(pass==1) //开锁状态时
{
if(second==1) //秒加到1s时
{
TR0=0; //关定时器
TL0=0xB0;
TH0=0x3C; //再次赋初值
second=0; //秒清零
}
}
else //不在开锁状态时
{
if(second==3) //秒加到3时
{
TR0=0; //关闭定时器
second=0; //秒清零
key_disable=0; //锁定键盘清零
s3_keydown=0;
TL0=0xB0;
TH0=0x3C; //重新赋初值
}
else
TR0=1; //打开定时器
}

}
}


📚程序源码和仿真资源

✨本案例仿真基于Proteus8.12平台

1
2
3
链接:https://pan.baidu.com/s/1RHjtnLqBrkVFwa4IcjpyEQ 
提取码:7vk1