单片机课程设计—万年历[1]郑州轻工业学院软件学院单片机与接口技术课程设计总结报告设计题目:电子万年历学生姓名:系别:专业:班级:学号:指导教师:2011年12月16日逻辑总框图:该电子万年历的总体设计框图如图(1)所示。
设计所需的元件:元件名称型号数量/个单片机 AT89C52 1时钟芯片 DS1302 1晶振 12MHz 1晶振 32.768kHz 1电容 30pF 2电容 22uF 1按键开关 4电阻 10K 9滑动变阻器 1K 1电池 1.5V 4LCD LCD1602 1电源Vcc +5V 1导线若干单元电路设计:1、主控制系统单片机中央处理系统的方案设计,选用AT89C52单片机作为中央处理器,如图(2)所示。
该单片机除了拥有MCS-51系列单片机的所有优点外,内部还具有8K的在系统可编程FLASH存储器,低功耗的空闲和掉电模式,极大的降低了电路的功耗,还包含了定时器、程序存储器、数据存储器等硬件,其硬件能符合整个控制系统的要求,不需要外接其他存储器芯片和定时器件,方便地构成一个最小系统。
整个系统结构紧凑,抗干扰能力强,性价比高。
2、时钟振荡电路时钟振荡电路图(3)所示,时钟振荡电路用于产生单片机正常工作时所需要的时钟信号,电路由两个30pF的瓷片电容和一个12MHz的晶振组成,并接入到单片机的XTAL1和XTAL2引脚处,使单片机工作于内部振荡模式。
此电路在加电后延迟大约10ms振荡器起振,在XTAL2引脚产生幅度为3V左右的正弦波时钟信号,其振荡频率主要由石英晶振的频率决定。
电路中两个电容C1、C2的作用使电路快速起振,提高电路的运行速度。
图(3)时钟振荡电路图图(4)复位电路3、复位电路复位电路由电阻和极性电容组成,如图(4)所示,通过高电平使单片机复位,在时钟电路开始工作后,当高电平的时间超过大约2us时,即可实现复位。
此复位电路为上电复位,较为简单。
若改进可以添加手动复位的功能,上电复位发生在开机加电时,由系统自动完成,手动复位通过一个按键来实现,在程序运行时,若遇到死机,死循环或程序“跑飞”等情况,通过手动复位就可以实现重新启动的操作。
手动按钮复位需要人为在复位输入端RST上加入高电平。
一般采用的办法是在RST端和正电源Vcc之间接一个按钮和一个电阻。
4、DS1302时钟电路时钟电路主要由时钟芯片DS1302、备用电池、晶振等几部分组成,如图(6)所示。
DS1302采用3线串行接口,占用引脚少,内部集成了可编程日历时钟,用户可以根据需要通过单片机的控制来自行设置,支持双电源供电,可以使用外部主电源和备用电源,备份电源能够使时钟芯片继续工作。
图(5) DS1302管脚图图(6) DS1302时钟电路DS1302各引脚的功能为:8: Vcc1:备用电池端;1: Vcc2:5V电源。
当Vcc2>Vcc1+0.2V时,由Vcc2向DS1302供电,当Vcc2< Vcc1时,由Vcc1向DS1302供电;7: SCLK:串行时钟,输入;6: I/O:数据输入输出口;5: CE/RST:复位脚;2、3: X1、X2 是外接晶振脚(32.768KHZ的晶振);4: 地(GND)。
DS1302有关日历、时间的寄存器:图(7)DS1302有关日历、时间的寄存器1、秒寄存器(81h、80h)的位7定义为时钟暂停标志(CH)。
当初始上电时该位置为1,时钟振荡器停止,DS1302处于低功耗状态;只有将秒寄器的该位置改写为0时,时钟才能开始运行。
2、小时寄存器(85h、84h)的位7用于定义DS1302是运行于12小时模式还是24小时模式。
当为高时,选择12小时模式。
在12小时模式时,位5是,当为1时,表示PM。
在24小时模式时,位5是第二个10小时位3、控制寄存器(8Fh、8Eh)的位7是写保护位(WP),其它7位均置为0。
在对任何的时钟和RAM的写操作之前,WP位必须为0。
当WP位为1时,写保护位防止对任一寄存器的写操作。
也就是说在电路上电的初始态WP是1,这时是不能改写上面任何一个时间寄存器的,只有首先将WP改写为0,才能进行其它寄存器的写操作。
DS1302读写时序DS1302是SPI总线驱动方式。
它不仅要向寄存器写入控制字,还需要读取相应寄存器的数据。
DS1302的控制字如图(8):图(8)DS1302的控制字图控制字的最高有效位(位7)必须是逻辑1,如果它为0,则不能把数据写入到DS1302中。
位6:如果为0,则表示存取日历时钟数据,为1表示存取RAM数据;位5至位1(A4~A0):指示操作单元的地址;位0(最低有效位):如为0,表示要进行写操作,为1表示进行读操作。
读数据:读数据时在紧跟8位的控制字指令后的下一个SCLK脉冲的下降沿,读出DS1302的数据,读出的数据是从最低位到最高位。
写数据:控制字总是从最低位开始输出。
在控制字指令输入后的下一个SCLK时钟的上升沿时,数据被写入DS1302,数据输入也是从最低位(0位)开始。
5、按键电路按键电路由四个轻触开关组成,如图(9)所示。
按键用来调整时间,其一端直接接到单片机的端口,另一端接地,当按下按键时,相应的端口变为低电平,通过一个与门只要这四个按键有一个按下就会在P3.2检测到一低电平就触发外部中断0进入按键调节程序中,通过与个各键相连的端口P3.4_P3.7可以判断是哪个键按下,从而作相应的操作。
图(9) 按键电路6、显示电路1602液晶也叫1602字符型液晶它是一种专门用来显示字母、数字、符号等的点阵型液晶模块它有若干个5X7或者5X11等点阵字符位组成,每个点阵字符位都可以显示一个字符。
显示电路采用LCD1602液晶显示,如图(10)所示,图中只画出了其相应的接口,3脚用于调节LCD1602的背光,4、5、6为LCD1602的控制口,用于控制其写入或是读出指令,7至14脚为LCD1602的数据口,将数传送到LCD1602中。
图(10) LCD1602显示电路LCD1602的特性+5V电压,对比度可调;内含复位电路;提供各种控制命令,如:清屏、字符闪烁、光标闪烁、显示移位等多种功能;有80字节显示数据存储器DDRAM;内建有160个5X7点阵的字型的字符发生器CGROM,8个可由用户自定义的5X7的字符发生器CGRAM;基本操作时序:读状态:输入:RS=L,RW=H,E=H;输出:DB0~DB7=状态字;写指令:输入:RS=L,RW=L,E=下降沿脉冲,DB0~DB7=指令码;输出:无。
读数据:输入:RS=H,RW=H,E=H;输出:DB0~DB7=数据;写数据:输入:RS=H,RW=L,E=下降沿脉冲,DB0~DB7=数据;输出:无。
LCD1602的各种指令不再一一说明。
流程图与软件设计:1、程序流程图主程序首先初始化定时器、LCD1602及DS1302,然后就开始查询按键,有键按下则开始调整时间和日期,若没有按下,则执行下面的时间、日期的显示,最后依次循环这些相同的操作,相应流程图如图(11)所示:图(12)程序流程图按键的检测是通过中断的办法来实现,利用按键进行间调整。
K1按下则开始设置时间及日期,同时在第一行最右端显示被选择的对象,第一次按下K1时,设置年份,若按下K3,则是减1操作,按下K2是加1操作,设置好年后,第二次按下K1时,则是设置月份,按K3减,按K2则加1,依次循环下去,则可以将时间和日期设置完毕,K4是确定键,设置好按下即可保存设置了。
2、软件设计软件总设计:主程序首先对系统环境初始化,设置定时器T0工作模式为16位定时/计数器模式,置位总中断允许位EA,并对键盘端口置位,再对LCD1602初始化,DS1302初始化。
接着扫描键盘,在键盘程序里面是对时间、日期及闹钟的调整,最下面是时间的显示。
软件程序编写:软件程序编写的好坏直接影响着系统运行情况的良好。
因本程序涉及的模块较多,所以程序编写也采用模块化设计,C语言具有编写灵活、移植方便、便于模块化设计的特点,所以本系统的软件采用C51编写。
具体程序见附件一:程序3、软件调试在软件调试过程中,当调节时间和日期后,单片机上电后更新的是PC的时间,后来查找资料发现,是设置ds1302的问题,对于开发板上的液晶一般RW都接的地,故不需要读液晶状态,也不需要读忙,但在仿真中还是加上了这一部分。
还有一个问题,在按键操作时有时会出现功能不稳定,这是由于按键存在抖动,所以后来加个去抖动的延时后在判断,基本就可以解决问题,整体电路与仿真结果分析:电子万年历硬件电路图及仿真如图(13)所示,系统由AT89C52单片机,按键扫描电路、显示电路、时钟电路、晶振电路、复位电路及电源指示电路。
仿真正确显示了时间,在LCD1602中正确显示了当前日期、时间,通过按按键K1,就可以开始设置时间,依次按K1依次在年、月、日、时、分之间切换,,按K2键用于加1操作,K3键用于减1操作,K4是确定按钮。
仿真正确显示了时间和日期,符合设计的要求。
图(13)电子万年历硬件电路图结论与心得:在这学期的课程序设计中,收获知识的同时,还收获了阅历,收获了成熟,通过查找大量资料,请教老师,以及不懈的努力,不仅培养了独立思考、动手制作的能力,在各种其它能力上也都有了提高。
更重要的是,在课程序设计里,我们学会了很多学习的方法,知道了理论和实践的巨大差别。
而这是以后最实用的,真的是受益匪浅。
要面对社会的挑战,只有不断的学习、实践,再学习、再实践。
同时在与老师和同学的交流过程中,互动学习,将知识融会贯通。
通过自己的努力,做出了一个万年历,对以后的学习是一个莫大的鼓舞,激起了我的学习兴趣和开发创新思维。
参考文献图书类:[1] 张毅坤陈善久,单片微型计算机原理及应用西安电子科技大学出版社[2] 张毅刚,,彭喜元,单片机原理与应用设计电子工业出版社[3] 赵建领薛园园,零基础学单片机C 语言程序设计机械工业出版社[4] 周向红 51单片机课程设计华中科技大学出版社,[5] 郭天祥 51单片机C语言教程-入门,提高,开发,拓展全攻略, 电子工业出版社[6] 赵亮侯国锐. 单片机C语言编程与实例人民邮电出版社附实验源程序:#include <reg51.h>#include <intrins.h>#include <string.h>#define uint unsigned int#define uchar unsigned charsbit IO= P1^0; //DS1302数据线sbit SCLK = P1^1; //DS130时钟线sbit RST = P1^2; //DS1302复位线sbit RS = P2^0; //LCD数据/命令选择端sbit RW = P2^1; //LCD读/写控制sbit EN = P2^2; //LCD使能端sbit K1=P3^4; //选择sbit K2=P3^5; //加sbit K3=P3^6; //减sbit K4=P3^7; //确定uchar tCount=0;uchar MonthsDays[]={0,31,0,31,30,31,30,31,31,30,31,30,31};uchar *WEEK[]={"SUN","MON","TUS","WEN","THU","FRI","SAT"};uchar LCD_DSY_BUFFER1[]={"DATE 00-00-00 "}; //显示格式uchar LCD_DSY_BUFFER2[]={"TIME 00:00:00 "};uchar DateTime[7]; //所读取的日期时间char Adjust_Index=-1; //当前调节的时间对象:,,分,是,日,月,年(1,2,3,4,6)uchar Change_Flag[]= "-MHDM-Y"; //(分,时,日,月,年)(不调节秒与周)/*---------延时程序------------------*/void DelayMS(uint ms){u char i;while(ms--){for(i=0;i<120;i++);}}//-----------向DS1302写入一字节------------------// void Write_A_Byte_TO_DS1302(uchar x){ u char i;for(i=0;i<8;i++){IO=x&0x01; //每一位与1与存入IO中SCLK=1;SCLK=0; //一个高脉冲将数据送入液晶控制器x>>=1; // 右移}}//-----------从DS1302读取一字节------------------// uchar Get_A_Byte_FROM_DS1302(){ u char i,b=0x00;for(i=0;i<8;i++){b |= _crol_((uchar)IO,i);SCLK=1;SCLK=0; //每一个高脉冲读取一位数据 }return b/16*10+b%16; //返回BCD码}//-----------从DS1302指定位置读数据------------------// uchar Read_Data(uchar addr){u char dat;RST = 0;SCLK=0;RST=1; //RST高电平时读/写Write_A_Byte_TO_DS1302(addr); //先写入地址dat = Get_A_Byte_FROM_DS1302();SCLK=1;RST=0;return dat;}//---------向DS1302某地址写入数据--------------------//void Write_DS1302(uchar addr,uchar dat){ S CLK=0;RST=1;Write_A_Byte_TO_DS1302(addr);Write_A_Byte_TO_DS1302(dat);SCLK=0;RST=0; //高脉冲写入数据}//--------------设置时间----------------//void SET_DS1302(){ u char i;//写控制字,取消写保护Write_DS1302(0x8E,0x00);//分时日月年依次写入for(i=1;i<7;i++){ //分的起始地址10000010(0x82),后面依次是时,日,月,周,年,写入地址每次递增2Write_DS1302(0x80+2*i,(DateTime[i]/10<<4)|(DateTime[i]%10));}Write_DS1302(0x8E,0x80); //加保护}//----------读取当前日期时间------------//void GetTime(){uchar i;for(i=0;i<7;i++){ DateTime[i]=Read_Data(0X81+2*i);} }//-----------读LCD状态------------------//uchar Read_LCD_State(){ u char state;RS=0;RW=1;EN=1; //输出:D0~D7=状态字DelayMS(1);state=P0; //从P0口读LCD状态EN = 0;DelayMS(1);return state;}//-----------忙等待------------------//void LCD_Busy_Wait(){w hile((Read_LCD_State()&0x80)==0x80);DelayMS(5);}//-----------向LCD写数据------------------//void Write_LCD_Data(uchar dat){L CD_Busy_Wait();RS=1;EN=0;RW=0; //写数据,EN为高脉冲,P0=dat;EN=1;DelayMS(1);EN=0;}//-------------写LCD指令-------------------//void Write_LCD_Command(uchar cmd){L CD_Busy_Wait();RS=0;EN=0; RW=0; //写指令,EN高脉冲,输出:D0~D7=数据P0=cmd;EN=1;DelayMS(1);EN=0;}//-------------LCD初始化-------------------//void Init_LCD(){W rite_LCD_Command(0x38); //设置16*2显示,5*7点阵,8位数据接口 DelayMS(1);Write_LCD_Command(0x01); //显示清零,数据指针清零DelayMS(1);Write_LCD_Command(0x06); //写一个字符后地址指针自动加1DelayMS(1);Write_LCD_Command(0x0c); //设置开显示,不显示光标DelayMS(1);}//------------------------------------------//设置液晶显示位置//------------------------------------------void Set_LCD_POS(uchar p){Write_LCD_Command(p|0x80);//相当于在0x80基础上加入位置量}//----在LCD上显示字符串---------//void Display_LCD_String(uchar p,uchar *s){ uchar i;Set_LCD_POS(p);for(i=0;i<16;i++){Write_LCD_Data(s[i]); //在固定位置显示时间日期DelayMS(1);}}//---------日期与时间值转换为数字字符----------------// void Format_DateTime(uchar d,uchar *a){a[0]=d/10+'0';a[1]=d%10+'0';}//判断是否为闰年uchar isLeapYear(uint y){ r eturn (y%4==0&&y%100!=0)||(y%400==0);}//求自2000.1.1开始的任何一天是星期几//函数没有通过,求出总天数后再求星期几//因为求总天数可能会超出uint的范围void RefreshWeekDay(){ u int i,d,w=5; //已知1999.12.31是周五for(i=2000;i<2000+DateTime[6];i++){d=isLeapYear(i)?366:365;w=(w+d)%7;}d=0;for(i=1;i<DateTime[4];i++){ d+=MonthsDays[i]; }d+=DateTime[3];//保存星期,0~6表示星期日,星期一,二,...,六,为了与DS1302的星期格式匹配,返回值需要加1DateTime[5]=(w+d)%7+1;}//*****年月日时分++/--********//void DateTime_Adjust(char x){ switch(Adjust_Index){case 6: //年00-99if(x==1&&DateTime[6]<99) DateTime[6]++;if(x==-1&&DateTime[6]>0) DateTime[6]--;//获取2月天数MonthsDays[2]=isLeapYear(2000+DateTime[6])?29:28;//如果年份变化后当前月份的天数大于上限则设为上限if(DateTime[3]>MonthsDays[DateTime[4]]){ DateTime[3]=MonthsDays[DateTime[4]];}RefreshWeekDay(); //刷新星期break;case 4: //月01-12if(x==1&&DateTime[4]<12) DateTime[4]++;if(x==-1&&DateTime[4]>1) DateTime[4]--;MonthsDays[2]=isLeapYear(2000+DateTime[6])?29:28;if(DateTime[3]>MonthsDays[DateTime[4]]){ DateTime[3]=MonthsDays[DateTime[4]];}RefreshWeekDay();break;case 3: //日00-28、29、30、31,调节之前首先根据年份得出该年中断二月天数MonthsDays[2]=isLeapYear(2000+DateTime[6])?29:28;//根据当前月份决定调节日期的上限if(x==1&&DateTime[3]<MonthsDays[DateTime[4]]) DateTime[3]++;if(x==-1&&DateTime[3]>0) DateTime[3]--;RefreshWeekDay();break;case 2: //时if(x==1&&DateTime[2]<23) DateTime[2]++;if(x==-1&&DateTime[2]>0) DateTime[2]--;break;case 1: //分if(x==1&&DateTime[1]<59) DateTime[1]++;if(x==-1&&DateTime[1]>0) DateTime[1]--;break;}//---定时器0每秒刷新LCD显示----//void T0_INT() interrupt 1{TH0=-50000/256;TL0=-50000%256;if(++tCount !=2) return;tCount=0;//按指定格式生成待显示的日期时间串Format_DateTime(DateTime[6],LCD_DSY_BUFFER1+5); Format_DateTime(DateTime[4],LCD_DSY_BUFFER1+8); Format_DateTime(DateTime[3],LCD_DSY_BUFFER1+11); //星期strcpy(LCD_DSY_BUFFER1+13,WEEK[DateTime[5]-1]); //时分秒Format_DateTime(DateTime[2],LCD_DSY_BUFFER2+5); Format_DateTime(DateTime[1],LCD_DSY_BUFFER2+8); Format_DateTime(DateTime[0],LCD_DSY_BUFFER2+11); //显示年月日,星期,时分秒Display_LCD_String(0x00,LCD_DSY_BUFFER1);Display_LCD_String(0x40,LCD_DSY_BUFFER2);}//----------键盘中断(INT0)-------------//void EX_INT0() interrupt 0{if(K1==0) //选择调整对象(Y M D H M)DelayMS(10);if(K1==0){//while(K1==0);if(Adjust_Index==-1||Adjust_Index==1){Adjust_Index=7;}Adjust_Index--;if(Adjust_Index==5) Adjust_Index=4;LCD_DSY_BUFFER2[13]='[';LCD_DSY_BUFFER2[14]=Change_Flag[Adjust_Index]; //显示调节对象LCD_DSY_BUFFER2[15]=']';}}else if(K2==0) //加{ //while(K2==0);DelayMS(10);if(K2==0)DateTime_Adjust(1);}else if(K3==0) //减{DelayMS(10);//while(K3==0);if(K3==0)DateTime_Adjust(-1);else if(K4==0) //确定{//while(K4==0);DelayMS(10);if(K4==0){SET_DS1302(); //将调整后的时间写入DS1302LCD_DSY_BUFFER2[13]=' ';LCD_DSY_BUFFER2[14]=' ';LCD_DSY_BUFFER2[15]=' ';Adjust_Index=-1;}}}void main(){ I nit_LCD(); //液晶初始化IE=0x83; //允许INT0,T0中断,EA=1,,ET0=1,EX0=1 IP=0x01; //设置外部中断0为高级中断IT0=0x01; //外部中断0为电平触发,低电平有效TMOD=0x01; //设置定时器T0工作方式为方式1,TH0=-50000/256; //装入初始值,定时1秒TL0=-50000%256;TR0=1; //启动定时器while(1){//如果未执行调整操作则正常读取当前时间if(Adjust_Index==-1) GetTime(); }}。