7:
SCLK:
串行时钟,输入;
6:
IO:
数据输入输出口;
5:
CERST:
复位脚;
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}
从DS1302指定位置读数据------
ucharRead_Data(ucharaddr)
{
uchardat;
RST=0;SCLK=0;RST=1;RST高电平时读写
Write_A_Byte_TO_DS1302(addr);先写入地址
dat=Get_A_Byte_FROM_DS1302();
SCLK=1;RST=0;
returndat;
}
向DS1302某地址写入数据
voidWrite_DS1302(ucharaddr,uchardat)
{SCLK=0;RST=1;
Write_A_Byte_TO_DS1302(addr);
Write_A_Byte_TO_DS1302(dat);
SCLK=0;RST=0;高脉冲写入数据
}
--设置时间----
voidSET_DS1302()
{uchari;
写控制字,取消写保护
Write_DS1302(0x8E,0x00);
分时日月年依次写入
for(i=1;i<7;i++)
{分的起始地址(0x82),后面依次是时,日,月,周,年,写入地址每次递增2
Write_DS1302(0x80+2*i,(DateTime[i]10<<4)|(DateTime[i]%10));
}
Write_DS1302(0x8E,0x80);加保护
}
读取当前日期时间
voidGetTime()
{uchari;
for(i=0;i<7;i++){DateTime[i]=Read_Data(0X81+2*i);}
}
读LCD状态------
ucharRead_LCD_State()
{ucharstate;
RS=0;RW=1;EN=1;输出:
D0~D7=状态字
DelayMS
(1);
state=P0;从P0口读LCD状态
EN=0;DelayMS
(1);
returnstate;
}
忙等待------
voidLCD_Busy_Wait()
{
while((Read_LCD_State()&0x80)==0x80);
DelayMS(5);
}
向LCD写数据------
voidWrite_LCD_Data(uchardat)
{
LCD_Busy_Wait();
RS=1;EN=0;RW=0;写数据,EN为高脉冲,
P0=dat;EN=1;DelayMS
(1);EN=0;
}
-写LCD指令
voidWrite_LCD_Command(ucharcmd)
{
LCD_Busy_Wait();
RS=0;EN=0;RW=0;写指令,EN高脉冲,输出:
D0~D7=数据
P0=cmd;EN=1;DelayMS
(1);EN=0;
}
-LCD初始化
voidInit_LCD()
{
Write_LCD_Command(0x38);设置16*2显示,5*7点阵,8位数据接口
DelayMS
(1);
Write_LCD_Command(0x01);显示清零,数据指针清零
DelayMS
(1);
Write_LCD_Command(0x06);写一个字符后地址指针自动加1
DelayMS
(1);
Write_LCD_Command(0x0c);设置开显示,不显示光标
DelayMS
(1);
}
------
设置液晶显示位置
------
voidSet_LCD_POS(ucharp){
Write_LCD_Command(p|0x80);相当于在0x80基础上加入位置量
}
----在LCD上显示字符串
voidDisplay_LCD_String(ucharp,uchar*s)
{uchari;
Set_LCD_POS(p);
for(i=0;i<16;i++)
{
Write_LCD_Data(s[i]);在固定位置显示时间日期
DelayMS
(1);
}
}
日期与时间值转换为数字字符----
voidFormat_DateTime(uchard,uchar*a)
{
a[0]=d10+'0';
a[1]=d%10+'0';
}
判断是否为闰年
ucharisLeapYear(uinty)
{return(y%4==0&&y%100!
=0)||(y%400==0);}
求自2000.1.1开始的任何一天是星期几
函数没有通过,求出总天数后再求星期几
因为求总天数可能会超出uint的范围
voidRefreshWeekDay()
{uinti,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{d+=MonthsDays[i];}
d+=DateTime[3];
保存星期,0~6表示星期日,星期一,二,...,六,为了与DS1302的星期格式匹配,返回值需要加1
DateTime[5]=(w+d)%7+1;
}
*****年月日时分++--********
voidDateTime_Adjust(charx)
{switch(Adjust_Index)
{
case6:
年00-99
if(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;
case4:
月01-12
if(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;
case3:
日00-28、29、30、31,调节之前首先根据年份得出该年中断二月天数
MonthsDays[2]=isLeapYear(2000+DateTime[6])?
29:
28;
根据当前月份决定调节日期的上限
if(x==1&&DateTime[3]if(x==-1&&DateTime[3]>0)DateTime[3]--;
RefreshWeekDay();
break;
case2:
时
if(x==1&&DateTime[2]<23)DateTime[2]++;
if(x==-1&&DateTime[2]>0)DateTime[2]--;
break;
case1:
分
if(x==1&&DateTime[1]<59)DateTime[1]++;
if(x==-1&&DateTime[1]>0)DateTime[1]--;
break;
}
}
---定时器0每秒刷新LCD显示----
voidT0_INT()interrupt1
{
TH0=;
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)-
voidEX_INT0()interrupt0
{
if(K1==0)选择调整对象(YMDHM)
{
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]=']';
}
}
elseif(K2==0)加
{while(K2==0);
DelayMS(10);
if(K2==0)
DateTime_Adjust
(1);
}
elseif(K3==0)减
{
DelayMS(10);while(K3==0);
if(K3==0)
DateTime_Adjust(-1);
}
elseif(K4==0)确定
{