如何用数码管实现iphone4S开机密码锁的程序

电子说

1.2w人已加入

描述

这一节要教会大家四个知识点:第一个:类似手机上10秒钟内无按键操作将自动进入锁屏的程序。第二个:如何用一个数组来接收按键的一串数字输入。第三个:矩阵键盘中,数字按键的输入,由于这部分按键的代码相似度非常高,因此把它封装在一个函数里可以非常简洁方便。第四个:继续加深熟悉鸿哥首次提出的“一二级菜单显示理论”:凡是人机界面显示,不管是数码管还是液晶屏,都可以把显示的内容分成不同的窗口来显示,每个显示的窗口中又可以分成不同的局部显示。其中窗口就是一级菜单,用ucWd变量表示。局部就是二级菜单,用ucPart来表示。不同的窗口,会有不同的更新显示变量ucWdXUpdate来对应,表示整屏全部更新显示。不同的局部,也会有不同的更新显示变量ucWdXPartYUpdate来对应,表示局部更新显示。具体内容,请看源代码讲解。(1)硬件平台:基于朱兆祺51单片机学习板。数字1键对应S1键,数字2键对应S2键,数字3键对应S3键…. 数字9键对应S9键, 数字0键对应S10键。其他的按键不用。(2)实现功能:本程序有3个窗口。开机显示第1个密码登录框窗口“----”,在这个窗口下输入密码,如果密码等于”9922”表示密码正确,将会切换到第2个显示按键值的窗口。在窗口2下,按不同的按键会显示不同的按键值,如果10秒内没有按键操作,将会自动切换到第1个密码登录窗口,类似手机上的自动锁屏操作。在密码登录窗口1下,如果密码不正确,会自动清除密码的数字,继续在窗口1下显示”----”。  窗口3是用来停留0.5秒显示全部密码的信息,然后根据密码的正确与否自动切换到对应的窗口。(3)源代码讲解如下:

#include "REG52.H"

#define const_no_key_push 4400   //大概10秒内无按键按下的时间

#define const_0_1s  220            //大概0.5秒的时间

#define const_voice_short  40   //蜂鸣器短叫的持续时间

#define const_key_time  20    //按键去抖动延时的时间

void initial_myself();    

void initial_peripheral();

void delay_short(unsigned int uiDelayShort); 

void delay_long(unsigned int uiDelaylong);

//驱动数码管的74HC595

void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01);  

void display_drive(); //显示数码管字模的驱动函数

void display_service(); //显示的窗口菜单服务程序

//驱动LED的74HC595

void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01);

void T0_time();  //定时中断函数

void number_key_input(unsigned char ucWhichKey);  //由于数字按键的代码相似度高,因此封装在这个函数里

void key_service(); //按键服务的应用程序

void key_scan();//按键扫描函数 放在定时中断里

sbit key_sr1=P0^0; //第一行输入

sbit key_sr2=P0^1; //第二行输入

sbit key_sr3=P0^2; //第三行输入

sbit key_sr4=P0^3; //第四行输入

sbit key_dr1=P0^4; //第一列输出

sbit key_dr2=P0^5; //第二列输出

sbit key_dr3=P0^6; //第三列输出

sbit key_dr4=P0^7; //第四列输出

sbit beep_dr=P2^7; //蜂鸣器的驱动IO口

sbit led_dr=P3^5;  //作为中途暂停指示灯 亮的时候表示中途暂停

sbit dig_hc595_sh_dr=P2^0;     //数码管的74HC595程序

sbit dig_hc595_st_dr=P2^1;  

sbit dig_hc595_ds_dr=P2^2;  

sbit hc595_sh_dr=P2^3;    //LED灯的74HC595程序

sbit hc595_st_dr=P2^4;  

sbit hc595_ds_dr=P2^5;  

unsigned char ucKeyStep=1;  //按键扫描步骤变量

unsigned int  uiKeyTimeCnt=0; //按键去抖动延时计数器

unsigned char ucKeyLock=0; //按键触发后自锁的变量标志

unsigned char ucRowRecord=1; //记录当前扫描到第几列了

unsigned char ucKeySec=0;   //被触发的按键编号

unsigned int  uiVoiceCnt=0;  //蜂鸣器鸣叫的持续时间计数器

unsigned char ucDigShow8;  //第8位数码管要显示的内容

unsigned char ucDigShow7;  //第7位数码管要显示的内容

unsigned char ucDigShow6;  //第6位数码管要显示的内容

unsigned char ucDigShow5;  //第5位数码管要显示的内容

unsigned char ucDigShow4;  //第4位数码管要显示的内容

unsigned char ucDigShow3;  //第3位数码管要显示的内容

unsigned char ucDigShow2;  //第2位数码管要显示的内容

unsigned char ucDigShow1;  //第1位数码管要显示的内容

unsigned char ucDigDot8;  //数码管8的小数点是否显示的标志

unsigned char ucDigDot7;  //数码管7的小数点是否显示的标志

unsigned char ucDigDot6;  //数码管6的小数点是否显示的标志

unsigned char ucDigDot5;  //数码管5的小数点是否显示的标志

unsigned char ucDigDot4;  //数码管4的小数点是否显示的标志

unsigned char ucDigDot3;  //数码管3的小数点是否显示的标志

unsigned char ucDigDot2;  //数码管2的小数点是否显示的标志

unsigned char ucDigDot1;  //数码管1的小数点是否显示的标志

unsigned char ucDigShowTemp=0; //临时中间变量

unsigned char ucDisplayDriveStep=1;  //动态扫描数码管的步骤变量

unsigned char ucWd1Update=1; //窗口1更新显示标志

unsigned char ucWd2Update=0; //窗口2更新显示标志

unsigned char ucWd3Update=0; //窗口3更新显示标志

unsigned char ucWd=1;  //本程序的核心变量,窗口显示变量。类似于一级菜单的变量。代表显示不同的窗口。

unsigned char ucInputPassword[4];  //在第1个窗口下,显示输入的4个密码

unsigned char ucPasswordCnt=0; //记录当前已经输入到哪一位密码了

unsigned char ucKeyNumber=1;  //在第2个窗口下,显示当前被按下的按键

unsigned int  uiNoKeyPushTimer=const_no_key_push;  //10秒内无按键按下的计时器

unsigned int  uiPasswordTimer=const_0_1s;  //显示0.5秒钟全部密码的计时器,让窗口3停留显示0.5秒钟之后自动消失

unsigned char ucTemp1=0;  //中间过渡变量

unsigned char ucTemp2=0;  //中间过渡变量

unsigned char ucTemp3=0;  //中间过渡变量

unsigned char ucTemp4=0;  //中间过渡变量

//根据原理图得出的共阴数码管字模表

code unsigned char dig_table[]=

{

0x3f,  //0       序号0

0x06,  //1       序号1

0x5b,  //2       序号2

0x4f,  //3       序号3

0x66,  //4       序号4

0x6d,  //5       序号5

0x7d,  //6       序号6

0x07,  //7       序号7

0x7f,  //8       序号8

0x6f,  //9       序号9

0x00,  //无      序号10

0x40,  //-       序号11

0x73,  //P       序号12

};

void main() 

{

initial_myself();  

delay_long(100);   

initial_peripheral(); 

while(1)  

key_service(); //按键服务的应用程序

display_service(); //显示的窗口菜单服务程序

}

}

/* 注释一:

*鸿哥首次提出的"一二级菜单显示理论":

*凡是人机界面显示,不管是数码管还是液晶屏,都可以把显示的内容分成不同的窗口来显示,

*每个显示的窗口中又可以分成不同的局部显示。其中窗口就是一级菜单,用ucWd变量表示。

*局部就是二级菜单,用ucPart来表示。不同的窗口,会有不同的更新显示变量ucWdXUpdate来对应,

*表示整屏全部更新显示。不同的局部,也会有不同的更新显示变量ucWdXPartYUpdate来对应,表示局部更新显示。

*/

void display_service() //显示的窗口菜单服务程序

{

switch(ucWd)  //本程序的核心变量,窗口显示变量。类似于一级菜单的变量。代表显示不同的窗口。

{

case 1:   //显示输入密码的登录框

if(ucWd1Update==1)  //窗口1要全部更新显示

{

ucWd1Update=0;  //及时清零标志,避免一直进来扫描

ucDigShow8=10;  //第8位数码管显示无

ucDigShow7=10;  //第7位数码管显示无

ucDigShow6=10;  //第6位数码管显示无

ucDigShow5=10;  //第5位数码管显示无

ucDigShow4=ucInputPassword[0];  //第4位数码管显示输入的密码

ucDigShow3=ucInputPassword[1];  //第3位数码管显示输入的密码

ucDigShow2=ucInputPassword[2];  //第2位数码管显示输入的密码

ucDigShow1=ucInputPassword[3];  //第1位数码管显示输入的密码

}

break;

case 2:  //显示被按下的键值

if(ucWd2Update==1)  //窗口2要全部更新显示

{

ucWd2Update=0;  //及时清零标志,避免一直进来扫描

ucDigShow8=10;  //第8位数码管显示无

ucDigShow7=10;  //第7位数码管显示无

ucDigShow6=10;  //第6位数码管显示无

ucDigShow5=10;  //第5位数码管显示无

ucDigShow4=10;  //第4位数码管显示无

ucDigShow3=10;  //第3位数码管显示无

ucDigShow2=10;  //第2位数码管显示无

ucDigShow1=ucKeyNumber; //第1位数码管显示被按下的键值

}

break;

case 3:   //当输入完4个密码后,显示1秒钟的密码登录框,

if(ucWd3Update==1)  //窗口3要全部更新显示

{

ucWd3Update=0;  //及时清零标志,避免一直进来扫描

ucDigShow8=10;  //第8位数码管显示无

ucDigShow7=10;  //第7位数码管显示无

ucDigShow6=10;  //第6位数码管显示无

ucDigShow5=10;  //第5位数码管显示无

ucDigShow4=ucInputPassword[0];  //第4位数码管显示输入的密码

ucDigShow3=ucInputPassword[1];  //第3位数码管显示输入的密码

ucDigShow2=ucInputPassword[2];  //第2位数码管显示输入的密码

ucDigShow1=ucInputPassword[3];  //第1位数码管显示输入的密码

}

break;     

}

}

void key_scan()//按键扫描函数 放在定时中断里

{  

switch(ucKeyStep)

{

case 1:   //按键扫描输出第ucRowRecord列低电平

if(ucRowRecord==1)  //第一列输出低电平

{

key_dr1=0;      

key_dr2=1;

key_dr3=1;    

key_dr4=1;

}

else if(ucRowRecord==2)  //第二列输出低电平

{

key_dr1=1;      

key_dr2=0;

key_dr3=1;    

key_dr4=1;

}

else if(ucRowRecord==3)  //第三列输出低电平

{

key_dr1=1;      

key_dr2=1;

key_dr3=0;    

key_dr4=1;

}

else   //第四列输出低电平

{

key_dr1=1;      

key_dr2=1;

key_dr3=1;    

key_dr4=0;

}

uiKeyTimeCnt=0;  //延时计数器清零

ucKeyStep++;     //切换到下一个运行步骤

break;

case 2:     //此处的小延时用来等待刚才列输出信号稳定,再判断输入信号。不是去抖动延时。

uiKeyTimeCnt++;

if(uiKeyTimeCnt>1)

{

uiKeyTimeCnt=0;

ucKeyStep++;     //切换到下一个运行步骤

}

break;

case 3:

if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==1)

{  

ucKeyStep=1;  //如果没有按键按下,返回到第一个运行步骤重新开始扫描

ucKeyLock=0;  //按键自锁标志清零

uiKeyTimeCnt=0; //按键去抖动延时计数器清零,此行非常巧妙     

ucRowRecord++;  //输出下一列

if(ucRowRecord>4)  

{

ucRowRecord=1; //依次输出完四列之后,继续从第一列开始输出低电平

}

}

else if(ucKeyLock==0)  //有按键按下,且是第一次触发

{

if(key_sr1==0&&key_sr2==1&&key_sr3==1&&key_sr4==1)

{

uiKeyTimeCnt++;  //去抖动延时计数器

if(uiKeyTimeCnt>const_key_time)

{

uiKeyTimeCnt=0;

ucKeyLock=1;//自锁按键置位,避免一直触发,只有松开按键,此标志位才会被清零

if(ucRowRecord==1)  //第一列输出低电平

{

ucKeySec=1;  //触发1号键 对应朱兆祺学习板的S1键

}

else if(ucRowRecord==2)  //第二列输出低电平

{

ucKeySec=2;  //触发2号键 对应朱兆祺学习板的S2键

}

else if(ucRowRecord==3)  //第三列输出低电平

{

ucKeySec=3;  //触发3号键 对应朱兆祺学习板的S3键

}

else   //第四列输出低电平

{

ucKeySec=4;  //触发4号键 对应朱兆祺学习板的S4键

}

}

}

else if(key_sr1==1&&key_sr2==0&&key_sr3==1&&key_sr4==1)

{

uiKeyTimeCnt++;  //去抖动延时计数器

if(uiKeyTimeCnt>const_key_time)

{

uiKeyTimeCnt=0;

ucKeyLock=1;//自锁按键置位,避免一直触发,只有松开按键,此标志位才会被清零

if(ucRowRecord==1)  //第一列输出低电平

{

ucKeySec=5;  //触发5号键 对应朱兆祺学习板的S5键

}

else if(ucRowRecord==2)  //第二列输出低电平

{

ucKeySec=6;  //触发6号键 对应朱兆祺学习板的S6键

}

else if(ucRowRecord==3)  //第三列输出低电平

{

ucKeySec=7;  //触发7号键 对应朱兆祺学习板的S7键

}

else   //第四列输出低电平

{

ucKeySec=8;  //触发8号键 对应朱兆祺学习板的S8键

}

}

}

else if(key_sr1==1&&key_sr2==1&&key_sr3==0&&key_sr4==1)

{

uiKeyTimeCnt++;  //去抖动延时计数器

if(uiKeyTimeCnt>const_key_time)

{

uiKeyTimeCnt=0;

ucKeyLock=1;//自锁按键置位,避免一直触发,只有松开按键,此标志位才会被清零

if(ucRowRecord==1)  //第一列输出低电平

{

ucKeySec=9;  //触发9号键 对应朱兆祺学习板的S9键

}

else if(ucRowRecord==2)  //第二列输出低电平

{

ucKeySec=10;  //触发10号键 对应朱兆祺学习板的S10键

}

else if(ucRowRecord==3)  //第三列输出低电平

{

ucKeySec=11;  //触发11号键 对应朱兆祺学习板的S11键

}

else   //第四列输出低电平

{

ucKeySec=12;  //触发12号键 对应朱兆祺学习板的S12键

}

}

}

else if(key_sr1==1&&key_sr2==1&&key_sr3==1&&key_sr4==0)

{

uiKeyTimeCnt++;  //去抖动延时计数器

if(uiKeyTimeCnt>const_key_time)

{

uiKeyTimeCnt=0;

ucKeyLock=1;//自锁按键置位,避免一直触发,只有松开按键,此标志位才会被清零

if(ucRowRecord==1)  //第一列输出低电平

{

ucKeySec=13;  //触发13号键 对应朱兆祺学习板的S13键

}

else if(ucRowRecord==2)  //第二列输出低电平

{

ucKeySec=14;  //触发14号键 对应朱兆祺学习板的S14键

}

else if(ucRowRecord==3)  //第三列输出低电平

{

ucKeySec=15;  //触发15号键 对应朱兆祺学习板的S15键

}

else   //第四列输出低电平

{

ucKeySec=16;  //触发16号键 对应朱兆祺学习板的S16键

}

}

}

}

break;

}

}

void key_service() //第三区 按键服务的应用程序

{

switch(ucKeySec) //按键服务状态切换

{

case 1:// 1号键 对应朱兆祺学习板的S1键

number_key_input(1);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;        

case 2:// 2号键 对应朱兆祺学习板的S2键

number_key_input(2);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;     

case 3:// 3号键 对应朱兆祺学习板的S3键

number_key_input(3);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;          

case 4:// 4号键 对应朱兆祺学习板的S4键

number_key_input(4);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 5:// 5号键 对应朱兆祺学习板的S5键

number_key_input(5);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 6:// 6号键 对应朱兆祺学习板的S6键

number_key_input(6);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 7:// 7号键 对应朱兆祺学习板的S7键

number_key_input(7);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 8:// 8号键 对应朱兆祺学习板的S8键

number_key_input(8);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 9:// 9号键 对应朱兆祺学习板的S9键

number_key_input(9);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 10:// 把这个按键专门用来输入数字0    对应朱兆祺学习板的S10键

number_key_input(0);  //由于数字按键的代码相似度高,因此把具体代码封装在这个函数里

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 11:// 11号键 对应朱兆祺学习板的S11键

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 12:// 12号键 对应朱兆祺学习板的S12键

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 13:// 13号键 对应朱兆祺学习板的S13键

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 14:// 14号键 对应朱兆祺学习板的S14键

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 15:// 15号键 对应朱兆祺学习板的S15键

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

case 16:// 16号键 对应朱兆祺学习板的S16键

uiVoiceCnt=const_voice_short; //按键声音触发,滴一声就停。

ucKeySec=0;  //响应按键服务处理程序后,按键编号清零,避免一致触发

break;   

}                

}

void number_key_input(unsigned char ucWhichKey)  //由于数字按键的代码相似度高,因此封装在这个函数里

{

switch(ucWd)

{

case 1:   //在显示密码登录框的窗口下

ucInputPassword[ucPasswordCnt]=ucWhichKey;   //输入的密码值显示

ucPasswordCnt++;

if(ucPasswordCnt>=4)

{

ucPasswordCnt=0;

ucWd=3;//切换到第3个的窗口,停留显示1秒钟全部密码

ucWd3Update=1;  //更新显示窗口3

uiPasswordTimer=const_0_1s;  //显示0.5秒钟全部密码的计时器,让窗口3停留显示0.5秒钟之后自动消失

}

ucWd1Update=1; //更新显示窗口1

uiNoKeyPushTimer=const_no_key_push;  //10秒内无按键按下的计时器赋新值

break;

case 2:   //在显示按键值的窗口下

ucKeyNumber=ucWhichKey; //输入的按键数值显示

ucWd2Update=1;  //更新显示窗口2

uiNoKeyPushTimer=const_no_key_push;  //10秒内无按键按下的计时器赋新值

break;

}

}

void display_drive()  

{

//以下程序,如果加一些数组和移位的元素,还可以压缩容量。但是鸿哥追求的不是容量,而是清晰的讲解思路

switch(ucDisplayDriveStep)

case 1:  //显示第1位

ucDigShowTemp=dig_table[ucDigShow1];

if(ucDigDot1==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfe);

break;

case 2:  //显示第2位

ucDigShowTemp=dig_table[ucDigShow2];

if(ucDigDot2==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfd);

break;

case 3:  //显示第3位

ucDigShowTemp=dig_table[ucDigShow3];

if(ucDigDot3==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xfb);

break;

case 4:  //显示第4位

ucDigShowTemp=dig_table[ucDigShow4];

if(ucDigDot4==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xf7);

break;

case 5:  //显示第5位

ucDigShowTemp=dig_table[ucDigShow5];

if(ucDigDot5==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xef);

break;

case 6:  //显示第6位

ucDigShowTemp=dig_table[ucDigShow6];

if(ucDigDot6==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xdf);

break;

case 7:  //显示第7位

ucDigShowTemp=dig_table[ucDigShow7];

if(ucDigDot7==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0xbf);

break;

case 8:  //显示第8位

ucDigShowTemp=dig_table[ucDigShow8];

if(ucDigDot8==1)

{

ucDigShowTemp=ucDigShowTemp|0x80;  //显示小数点

}

dig_hc595_drive(ucDigShowTemp,0x7f);

break;

}

ucDisplayDriveStep++;

if(ucDisplayDriveStep>8)  //扫描完8个数码管后,重新从第一个开始扫描

{

ucDisplayDriveStep=1;

}

}

//数码管的74HC595驱动函数

void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01)

{

unsigned char i;

unsigned char ucTempData;

dig_hc595_sh_dr=0;

dig_hc595_st_dr=0;

ucTempData=ucDigStatusTemp16_09;  //先送高8位

for(i=0;i<8;i++)

if(ucTempData>=0x80)dig_hc595_ds_dr=1;

else dig_hc595_ds_dr=0;

dig_hc595_sh_dr=0;     //SH引脚的上升沿把数据送入寄存器

delay_short(1); 

dig_hc595_sh_dr=1;

delay_short(1);

ucTempData=ucTempData<<1;

}

ucTempData=ucDigStatusTemp08_01;  //再先送低8位

for(i=0;i<8;i++)

if(ucTempData>=0x80)dig_hc595_ds_dr=1;

else dig_hc595_ds_dr=0;

dig_hc595_sh_dr=0;     //SH引脚的上升沿把数据送入寄存器

delay_short(1); 

dig_hc595_sh_dr=1;

delay_short(1);

ucTempData=ucTempData<<1;

}

dig_hc595_st_dr=0;  //ST引脚把两个寄存器的数据更新输出到74HC595的输出引脚上并且锁存起来

delay_short(1); 

dig_hc595_st_dr=1;

delay_short(1);

dig_hc595_sh_dr=0;    //拉低,抗干扰就增强

dig_hc595_st_dr=0;

dig_hc595_ds_dr=0;

}

//LED灯的74HC595驱动函数

void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01)

{

unsigned char i;

unsigned char ucTempData;

hc595_sh_dr=0;

hc595_st_dr=0;

ucTempData=ucLedStatusTemp16_09;  //先送高8位

for(i=0;i<8;i++)

if(ucTempData>=0x80)hc595_ds_dr=1;

else hc595_ds_dr=0;

hc595_sh_dr=0;     //SH引脚的上升沿把数据送入寄存器

delay_short(1); 

hc595_sh_dr=1;

delay_short(1);

ucTempData=ucTempData<<1;

}

ucTempData=ucLedStatusTemp08_01;  //再先送低8位

for(i=0;i<8;i++)

if(ucTempData>=0x80)hc595_ds_dr=1;

else hc595_ds_dr=0;

hc595_sh_dr=0;     //SH引脚的上升沿把数据送入寄存器

delay_short(1); 

hc595_sh_dr=1;

delay_short(1);

ucTempData=ucTempData<<1;

}

hc595_st_dr=0;  //ST引脚把两个寄存器的数据更新输出到74HC595的输出引脚上并且锁存起来

delay_short(1); 

hc595_st_dr=1;

delay_short(1);

hc595_sh_dr=0;    //拉低,抗干扰就增强

hc595_st_dr=0;

hc595_ds_dr=0;

}

void T0_time() interrupt 1

{

unsigned int i;

TF0=0;  //清除中断标志

TR0=0; //关中断

if(ucWd==3)  //在窗口3下

{

if(uiPasswordTimer>0)   

{

uiPasswordTimer--;  

}

if(uiPasswordTimer==0)

{

if(ucInputPassword[0]==9&&ucInputPassword[1]==9&&ucInputPassword[2]==2&&ucInputPassword[3]==2)  

{     //如果密码等于9922,则正确

ucWd=2;//切换到第2个显示按键的窗口

ucWd2Update=1;  //更新显示窗口2

}

else //如果密码不正确,则继续显示----

{

for(i=0;i<4;i++)

{

ucInputPassword[i]=11;  //开机默认密码全部显示"----"

}

ucWd=1; 

ucWd1Update=1; //更新显示窗口1

}

}

}

if(ucWd==2)  //在窗口2下

{

if(uiNoKeyPushTimer>0)   

{

uiNoKeyPushTimer--;  

}

if(uiNoKeyPushTimer==0)//如果10秒内无按键按下,则自动切换到显示密码登录框的界面

{

for(i=0;i<4;i++)

{

ucInputPassword[i]=11;  //开机默认密码全部显示"----"

}

ucWd=1; 

ucWd1Update=1; //更新显示窗口1

}

}

key_scan(); //按键扫描函数

if(uiVoiceCnt!=0)

{

uiVoiceCnt--; //每次进入定时中断都自减1,直到等于零为止。才停止鸣叫

beep_dr=0;  //蜂鸣器是PNP三极管控制,低电平就开始鸣叫。

//     beep_dr=1;  //蜂鸣器是PNP三极管控制,低电平就开始鸣叫。

}

else

{

; //此处多加一个空指令,想维持跟if括号语句的数量对称,都是两条指令。不加也可以。

beep_dr=1;  //蜂鸣器是PNP三极管控制,高电平就停止鸣叫。

//     beep_dr=0;  //蜂鸣器是PNP三极管控制,高电平就停止鸣叫。

}

display_drive();  //数码管字模的驱动函数

TH0=0xfe;   //重装初始值(65535-500)=65035=0xfe0b

TL0=0x0b;

TR0=1;  //开中断

}

void delay_short(unsigned int uiDelayShort) 

{

unsigned int i;  

for(i=0;i

{

;   //一个分号相当于执行一条空语句

}

}

void delay_long(unsigned int uiDelayLong)

{

unsigned int i;

unsigned int j;

for(i=0;i

{

for(j=0;j<500;j++)  //内嵌循环的空指令数量

{

; //一个分号相当于执行一条空语句

}

}

}

void initial_myself()  //第一区 初始化单片机

{

led_dr=0;  //关闭独立LED灯

beep_dr=1; //用PNP三极管控制蜂鸣器,输出高电平时不叫。

hc595_drive(0x00,0x00);  //关闭所有经过另外两个74HC595驱动的LED灯

TMOD=0x01;  //设置定时器0为工作方式1

TH0=0xfe;   //重装初始值(65535-500)=65035=0xfe0b

TL0=0x0b;

}

void initial_peripheral() //第二区 初始化外围

{

unsigned int i; //个人的变量命名习惯,i,j,k等单个字母的变量名只用在for循环里

for(i=0;i<4;i++)

{

ucInputPassword[i]=11;  //开机默认密码全部显示"----"

}

ucDigDot8=0;   //小数点全部不显示

ucDigDot7=0;  

ucDigDot6=0; 

ucDigDot5=0;  

ucDigDot4=0; 

ucDigDot3=0;  

ucDigDot2=0;

ucDigDot1=0;

EA=1;     //开总中断

ET0=1;    //允许定时中断

TR0=1;    //启动定时中断

}

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分