电子技术课程设计任务书
一.设计分组:
机械设计及自动化专业061、062班每两人一组
二.时间安排:
2周,进度要求如下
课程设计起止日期:20##年6月8 日—6月17日
三、课程设计任务(即要求):
课程设计题目:数字钟的设计与制作
(一)设计指标:
1.显示时、分、秒。采用24小时制。
2.制作、调试出一个具有直流电源、简易信号源及用来计“时”“分”“秒”的数字钟系统。并按照直流电源、简易信号源、及“秒”、“分”进位和“时”循环进位是否正常给予不同记分。
3.具有校时功能,可以对小时和分单独校时,对分校时的时候,停止分向小时进位。校时时钟源可以手动输入或借用电路中的时钟。
(二)具体要求:
1.设计方案的论证和选择
(1)、方案提出
*查阅资料确定数字钟的电路框图。
*提出两种以上数字钟的电路设计方案。
(2)、方案选择和论证
*分析比较,并学会用Multisim软件进行仿真。最后根据仿真结果,在认真论证的基础上,选择出可行性较强、性能可靠、成本合理,基本符合此次设计要求的最优方案。(要求有仿真波形图和仿真电路图)。
*确定设计方案后,用Protel(或Multisim)软件绘出其电原理图。
注意的问题:(1)应针对关系到电路全局的问题,多动脑筋,多提些不同的方案,深入分析比较,从而找出最优方案。(2)既考虑方案的可行性,还考虑性能、可靠性等实际问题。
2.制作、调试数字钟实物 ;要求焊接、调试出一个具有直流电源、简易信号源及用来计“时”“分”“秒”的数字钟系统。
3.按设计任务书的要求的格式,撰写或打印课程设计报告书。
4.设计总结和答辩。
(三)实验仪器、工具:
1. 共阳(共阴)七段数码管/计数器/译码驱动集成电路。
2. 导线/电阻/电容/石英晶体/变压器等。
3.示波器、万用表。
四.设计报告要求:
格式要求:(见附录)
内容要求:
1. 画出设计的原理框图,并要求说明该框图的工作过程及每个模块的功能。
2. 画出各功能模块的电路图,加以原理说明(如10进制到6进制转换的原理,个位到十位的进位信号选择和变换等)。
3.描述设计制作的数字钟及其运行结果。说明测试中出现的故障及其排除方法
4.总结:设计过程中遇到的问题及解决办法;课程设计中的心得体会;对课程设计内容、方式、要求等各方面的建议。
5.附录1:画出总布局接线图(集成块按实际布局位置画,关键的连接单独应画出,计数器到译码器的数据线、译码器到数码管的数据线可以简化画法,但集成块的引脚须按实际位置画,并注明名称。)(或附上实物照片)
6. 附录2:元器件清单。
五、课程设计参考资料
1.彭介华主编:《电子技术课程设计指导》,高等教育出版社,20##年出版。
2.郑步生. Multisim2001电路设计及仿真入门与应用.电子工业出版社.2002
3.高吉祥主编.电子技术基础实验与课程设计.北京:电子工业出版社,2002
4.扬志亮. Protel99SE电路原理图设计技术 .西北工业大学出版社. 2002
5.http://www.21ic.com/
6.http://www.cediy.com/
20##年6月2日
一.指标要求:
1.显示时、分、秒。采用24小时制。
2.制作、调试出一个具有直流电源、简易信号源及用来计“时”“分”“秒”的数字钟系统。并按照直流电源、简易信号源、及“秒”、“分”进位和“时”循环进位是否正常给予不同记分。
3.具有校时功能,可以对小时和分单独校时,对分校时的时候,停止分向小时进位。校时时钟源可以手动输入或借用电路中的时钟。
二.设计计算:
1.总体方案设计:
数字钟实际上是一个对标准频率(1HZ)进行计数的计数电路。由于计数的起始时间不可能与标准时间(如北京时间)一致,故需要在电路上加一个校时电路,同时标准的1HZ时间信号必须做到准确稳定。通常使用石英晶体振荡器电路构成数字钟。图 3-1所示为数字钟的一般构成框图。
图3-1 数字钟的组成框图
如图3-1所示,数字钟一般都由直流电源(图中未画出)、振荡器、分频器、计数器、译码器、显示器等几部分组成。其中,振荡器和分频器组成标准“秒”信号发生器,由不同进制的计数器、译码器和显示器组成计时系统。“秒”信号送入计数器进行计数,把累计的结果以“时”、“分”、“秒”的数字显示出来。“时”显示由二十四进制计数器、译码器、显示器构成,“分”、“秒”显示分别由六十进制计数器、译码器、显示器构成。
2.单元电路设计:
(1)直流稳压电源电路
给数字钟提供+5V稳定电流。如图3-2所示电路是由W7805集成稳压器组成的直流稳压电源。电源由220V交流电经变压器降压后再经桥式整流器整流,后经电容滤波和集成稳压器稳压后得到。图中集成稳压器输入端接至独立的整流电路的输出上,因而+5V输出电压的稳定性能很好。
图3-2 直流稳压电源电路
(2)晶体振荡器电路
给数字钟提供一个频率稳定准确的32768Hz的方波信号,可保证数字钟的走时准确及稳定。不管是指针式的电子钟还是数字显示的电子钟都使用了晶体振荡器电路。一般输出为方波的数字式晶体振荡器电路通常有两类,一类是用TTL门电路构成;CMOS非门构成的电路;一类是555构成,如图3-3所示。本次设计采用了后一种。如图3-4所示,由CMOS非门U1与晶体、电容和电阻构成晶体振荡器电路,U2实现整形功能,将振荡器输出的近似于正弦波的波形转换为较理想的方波。输出反馈电阻R1为非门提供偏置,使电路工作于放大区域,即非门的功能近似于一个高增益的反相放大器。电容C1、C2与晶体构成一个谐振型网络,完成对振荡频率的控制功能,同时提供了一个180度相移,从而和非门构成一个正反馈网络,实现了振荡器的功能。由于晶体具有较高的频率稳定性及准确性,从而保证了输出频率的稳定和准确。
晶体XTAL的频率选为32768HZ。该元件专为数字钟电路而设计,其频率较低,有利于减少分频器级数。
从有关手册中,可查得C1、C2均为30pF。当要求频率准确度和稳定度更高时,还可接入校正电容并采取温度补偿措施。
由于CMOS电路的输入阻抗极高,因此反馈电阻R1可选为10MΩ。较高的反馈电阻有利于提高振荡频率的稳定性。
图3-3 555振荡器 图3-4 COMS晶体振荡器
(3)分频器电路
通常,数字钟的晶体振荡器输出频率较高,为了得到1Hz的秒信号输入,需要对振荡器的输出信号进行分频。
所谓分频,就是脉冲频率每经一级触发器就减低一半,即周期增加一倍。由所学知识可知,一位二进制计数器是一个二分频器。以此类推,当二进制计数器有n位时,第n级触发器输出脉冲的频率是计数脉冲的1/2n。故实现分频的电路是计数器电路。如果石英晶体振荡器的振荡频率为32768Hz。则将32768Hz的振荡信号分频为1Hz的分频倍数为32768(215),即实现该分频功能的计数器相当于15级2进制计数器。
本实验中采用CD4060来构成分频电路。CD4060在数字集成电路中可实现的分频次数最高,而且CD4060还包含振荡电路所需的非门,使用更为方便。
CD4060计数为14级2进制计数器,可以将32768Hz的信号分频为2Hz,之后用JK触发器完成2分频,就可以得到标准1Hz脉冲。CD4060的内部框图如图3-5所示,从图中可以看出,CD4060的时钟输入端两个串接的非门,因此可以直接实现振荡和分频的功能。
图3-5 CD4046内部框图
(4)时间计数单元
时间计数单元有时计数、分计数和秒计数等几个部分。时计数单元一般为二十四进制计数器计数器,其输出为两位8424BCD码形式;分计数和秒计数单元为六十进制计数器,其输出也为8421BCD码。
采用异步十进制计数器74LS290来实现计数单元的计数功能。
先介绍下74LS290:
图3-6 74LS290管脚图
74LS290功能表
其中Q3Q2Q1Q0即为QDQCQBQA。
秒计数单元为六十进制计数器,它有两位组成。各位为十进制,无需进制转换,只需将QA与CPB(下降沿有效)相连即可。CPA与1Hz秒输入信号相连,QD可作为向上的进位信号与十位计数单元的CPA相连。十位为六进制,需要进制转换。电路连接如图3-7所示。其中QC可作为向上的进位信号与分个位的计数单元的CPA相连。个位十进制计数器经过十个脉冲循环一次,每当第十个脉冲来到时,QD由1变为0,相当于一个下降沿,使十位六进制计数器计数。各位计数器经过第一次十个脉冲,十位计数器计数为0001;经过二十个脉冲,计数为0010;以此类推,经过六十个脉冲,计数为0110。接着,立即清零,个位和十位计数器都恢复为0000。
图3-7 六十进制
分个位和分十位计数单元电路结构分别与秒个位和秒十位计数单元完全相同,只不过分个位计数单元的QD作为向上的进位信号应与分十位计数单元的CPA相连,分十位计数单元的QC作为向上的进位信号应与时个位计数单元的CPA相连。
时个位计数单元电路结构仍与秒或个位计数单元相同,但是要求整个时计数单元应为二十四进制计数器,不是10的整数倍,因此需将个位和十位计数单元合并为一个整体才能进行二十四进制转换。利用2片74LS290实现二十四进制计数功能的电路如图3-8所示。
图3-8 二十四进制
当时计数单元的个位计数器计数为0100,十位计数器计数为0010时,清零。
(5)译码驱动及显示单元
在数字钟电路中,译码器的输入信号就是计数器的输出信号,它的输出端接至数码管。计数器输出的四位BCD码经译码后,变成某个十进制数字对应的控制电平,去驱动数码管各段发光,从而把该数字显示出来。试验中采用8421BCD译码器74LS248。它与LED数码管的连接如图3-9所示。
图3-9共阴极译码电路
(6)校时电路
当刚接通电源或计时出现误差时,都需要对时间进行校正。因此,应截断分个位和时个位的直接计数通路,并采用正常计时信号与校正信号可以随时切换的电路接入其中。即为用TTL与非门实现的时或分校时电路,a端与低位的进位信号相连;b端与校正信号相连,校正信号直接取自分频器产生的1Hz信号;输出端则与分或时个位计时输入端相连。当开关打向右侧时,与非门F的一个输入端为1,将它开通,使计数器输出的脉冲加到F的另一输入端,并经H进入分或时计数器,而此时G由于一个输入端为0,因此被关断,校时用的脉冲进不去,故处于正常计时状态;当开关打向左侧时,情况正好与上述相反,F被封闭,G打开,这时校时电路处于自动校时状态。
实际使用时,因为电路开关存在抖动问题,所以一般会接一个基本RS触发器构成开关消抖动电路。所以整个校时电路就如图3-10所示。
图3-10 校时电路
3.总体电路:
总体电路如图4-1所示。
元器件清单:6片74LS290、6片74LS248、3片74LS00、6个LED数码管。
四.软件模拟
图4-1 数字钟的原理电路(方案一)
图4-2 数字钟的原理电路(方案二)
方案一在实现时借助与非逻辑门来实现,A、B分别是时校时、分校时开关,不校正时,A、B是闭合的。当校正“时”位时,需把A开关拨向另一边,就能使“时”位自动校时,校正完毕后把开关拨向另一边即可。“分”位的校正和“时”位的校正方法一样。另外校时电路中接有基本RS触发器构成的开关消抖动电路,消除了电路开关存在的抖动问题。
方案二其它电路与方案一相同,只是校时电路部分不同。此方案的校时电路直接有开关来实现,通过各开关的闭合来实现“时”位和“分”位的校时。本方案的缺点是需要手动把开关来回拨动一次才能实现计数一次,且电路消除不了电路开关存在的抖动问题。
五.安装调试
因仿真与实际元件上的差异,所以我们实际安装是按照另外一张图纸的设计电路进行的。基本上整个安装过程都比较顺利,我们先是把秒位的电路焊接上去,调试正常后,接着我们把分位的电路和时位的电路焊接上去。整个安装调试过程花了一天半的时间,直至一切正常。最终的实物图如图5-1所示
六.总结:
1.设计过程中遇到的问题及解决办法
在整个设计安装过程中,基本上一切都顺利,中途遇到了一个问题,就是七段LED数码管的显示问题。安装调试过程中,秒位一切正常,接着焊接分位,完成后,分位的十位数码管有一段不会亮,刚开始以为是数码管坏了,换后还是不能显示,接着把译码器换了,不行,再用万用表检测那段不亮的数码管对应的电阻,电阻正常,还是不行,最后发现是虚焊造成的,不亮的那段数码管和译码器之间有些接触不良。正常后接着继续接时位电路,完成后数码管完全没有显示数字,再次检查芯片、电阻和电路,一切正常,可数码管还是不亮,在老师的指导下,我用万用表检测了时位的各片芯片,最后发现有一片74LS00的电压显示不正常,于是我们检查了附近的电路,并把附近的电路进行了重焊,最后解决了数码管完全没有显示的问题,还是由电路接触不良造成的,可见确认芯片和电路是否接触良好是非常重要的一件事,尽管我们在焊接过程中非常仔细认真。
2.设计体会
在此次的数字钟设计过程中,我从中学到了不少东西,让我了解了设计电路的程序,也让我了解了关于数字钟的原理与设计理念,要设计一个电路总要先用仿真仿真成功之后才实际接线的,但是最后的成品却不一定与仿真时完全一样,因为,再实际接线中有着各种各样的条件制约着。另外我更加熟悉了芯片的结构,熟练掌握了各芯片的工作原理和其具体的使用方法以及任意进制计数器的改接。加强了理论知识与实践统一的能力,加强了自己的动手操作能力,为以后的电路设计打好基础。另外,焊接真是一门手艺活,为了把焊点焊得漂亮,连线整洁美观,我花了不少功夫。
3.对该设计的建议
对此次的设计,我比较满意,收获很大,学到了很多书本上学不到的东西,培养了自己查找资料解决问题的能力,希望学校能多安排这种课程设计,提高同学们的学习兴趣。
图5-1 实物图
七.参考文献:
[1] 秦曾煌.电工学 电子技术.北京:高等教育出版社,2004.
[2] 蒋卓勤. Multisim2001及其在电子设计中的应用.西安:西安电子科技大学出版社,2004
[3] 高有堂等.电子设计与实战指导.北京:电子工业出版社,2007
[4] 谢自美. 电子线路设计.实验.测试.武汉:华中科技大学出版社,2003
[5] 卢结成等.电子电路实验及应用课题设计.合肥:中国科学技术大学出版社,2006
第二篇:数字钟课程设计
Proteus与Keil C课程设计
题目:数字钟课程设计
院系:机电工程学院
专业:应用电子技术
班级:零九一班
姓名:
学号:20090615
指导老师:杨志邦
日期:2011.11.17
1 引言
单片机因将其主要组成部分集成在一个芯片上而得名,就是把中央处理器、随机存储器、只读存储器、中断系统、定时器/计数器以及I/O接口电路等部件集成在一个芯片上。
在我国,单片机的开发应用已有15年左右,已经形成一支庞大的技术开发队伍,为我国单片机应用积累了丰富的经验。随着电子技术、计算机芯片技术和微电子技术的飞速发展促进了单片机技术一日千里的变化。
随着半导体技术的飞速发展,以及移动通信、网络技术、多媒体技术在嵌入式系统设计中的应用,单片机从4位、8位、16位到32位,其发展历程一直受到广大电子爱好者的极大关注。单片机功能越来越强大,价格却不断下降的优势无疑成为嵌入式系统方案设计的首选,同时单片机应用领域的扩大也使得更多人加入到基于单片机系统的开发行列中,推动着单片机技术的创新进步。
2 1 整体设计思路
这部分主要介绍工作安排和整体设计的思想。工作过程规划如下:
图2.1 整体设计思路
针对要实现的功能,拟采用STC89C51单片机进行设计,STC89C51 单片机是一款低功耗,高性能CMOS8位单片机,片内含4KB在线可编程(ISP)的可反复擦写1000次的Flash只读程序存储器,器件采用高密度、非易失性存储技术制造,兼容标准MCS- 51指令系统及STC80C51引脚结构。这样,既能做到经济合理又能实现预期的功能。
在程序方面,采用分块设计的方法,这样既减小了编程难度、使程序易于理解,又能便于添加各项功能。程序可分为闹钟的声音程序、显示程序、闹钟显示程序、调时显示、定时程序。本电子钟设计主要是依照图2.1中的流程做出来的,时间分配比较均匀。首先,在编程之前必须了解硬件结构尤其是各引脚的用法,以及内部寄存器、存储单元的用法,否则,编程无从下手,电路也无法设计。这是前期准备工作。第二部分是硬件部分:依据想要的功能分块设计设计,比如输入需要开关电路,输出需要显示驱动电路和数码管电路等。第三部分是软件部分:先学习理解汇编语言的编程方法再根据设计的硬件电路进行分块的编程调试,最终完成程序设计。第四部分是软件画图部分:设计好电路后进行画图,包括电路图和仿真图的绘制。第五部分是软件仿真部分:软硬件设计好后将软件载入芯片中进行仿真,仿真无法完成时检查软件程序和硬件电路并进行修改直到仿真成功。第六部分是硬件实现部分:连接电路并导入程序检查电路,若与设计的完全一样一般能实现想要的功能。
2 主要元件的使用
下面就本次设计中用到的主要元件的所有功能进行简单的介绍,包括STC89C51单片机、74LS07芯片、以及数码管的特性和用法。
3 STC89C51单片机:
该单片机功能强大,不仅能满足设计的需要,也可以在设计要求的基础上进行一些扩展。单片机的结构如下:
图3.1.1 单片机引脚图
在使用时VCC接电源电压,GND接地。P0,P1,P2,P3可作为输入或输出端口,RST是复位输入,接复位电路。XTAL1和XTAL2接复位电路。这些可以在硬件设计部分体现出来。
4 电路设计
4.1整体设计
此次设计主要是应用单片机来设计电子时钟,硬件部分主要分以下电路模块:显示电路用六个数码管分别显示小时(年份)、分钟(月份)和秒(日),通过动态扫描进行显示,从而避免了译码器的使用,使电路更加简单。单片机采用AT89C51系列,这种单片机应用简单,适合电子钟设计。
电路的总体设计框架如下:
图4.1.1 电路模块图
4.2 分块设计
这部分介绍各模块电路的设计方法和成果,主要分为:输入部分、输出部分、复位和晶振电路。
4.2.1 输入部分
在电子钟的输入部分,设置相应的置数功能,通过外部设备的输入,如按键,实现时间的修改。除此之外,调整闹铃、定时、日期时也需要按键进行输入。在选用输入端口时,将P3引脚与按键相连进行输入。
设计的输入部分如下:
图4.2.1 输入部分
4.2.2 输出部分(显示电路)
该部分电路图如下所示:
图4.2.2 显示部分
在实际电路中采用LCD 显示屏相连进行显示,先把LCD 显示屏的7、8、9、10、11、12、13、14与RP8对应2~9相连,然后把LCD 显示屏的2和15引脚各自相连电源,1接地,3,15分别连接热敏电阻及晶体管的集电极。
4.2.3 晶振与复位电路:
图4.2.3 晶振与复位电路
5程序设计5.1程序思路
图5.1.1 程序设计思路
结合电路,程序的总体思路是:
1、点复位键后,进行时间显示,从0时0分0秒开始。
2、按下按键1时,进行调时,此时按下4调整时,按下5调整分,若2秒钟未按键,则不再等待,恢复走时,持续按键时大约0.3秒步进1,下同。
3、按下2键时进行闹铃调整,用4、5键分别调时和分,此时只有前四位进行显示,即闹铃功能精确到分,2秒钟无按键则返回时间显示,时间到达闹铃所定时间时P0.7输出高电平,蜂鸣器响,按下按键2或3时蜂鸣器停止。
4、按3键时进行定时的设定,同样,分别通过4、5调整分和秒,若两秒未按键则不进行定时,设定之后再次按下3键则进行倒计时,倒计时时间未到时若按下1键则进行时间显示;若倒计时时间到则P0.7为高电平,蜂鸣器响。定时的最大值为59分59秒。
在程序设计时,尽量改进算法,算法的改进可以使相对误差减小,或者可以使占用空间减小。另外,分块的设计思想要贯穿始终,整个程序较为繁杂,某些程序段会反复用到,因此采取的方法是写出多个程序段,通过跳转指令进行调用。
5.2程序设计步骤
在程序设计过程中,我遇到了很多困难,这部分也是让我学到很多东西的地方。
首先,我学习了定时器的相关知识,计数器的使用是很重要的组成部分,在这个设计中选择计数器T0。T0的工作方式有:
方式0:不推荐
方式1:16位计数器,常用
方式2:自动重装初值的8位定时/计数器
方式3:T0相当于两个独立的8位定时/计数器
此程序采用方式1,方式1的定时时间t为t=(216-M)*12/fosc。其中M为定时器初值,fosc为12MHz,若M为0则t=65536*12/2*106=65.536ms。因此可取50ms为计时单位,初值M应为50*10-3*106=216-M。M=15536=11110010110000=3CB0。即定时器初值为TH0=03CH,TL0=0B0H。定时器中断20次为一秒,这部分在中断程序中用到。
5.3程序的主要模块
5.3.1延迟程序
在动态扫描时,必然用到延迟程序,这里使用延迟1ms的程序,此程序需要反复调用。此段程序是很简单的,但就是在这段简单的程序上,也会出现问题,
5.3.2 中断服务子程序:
中断服务程序中,总体思路是:由于初值是3CB0H,所以装满定时器需要50ms的时间,从而20次中断为一秒,一秒之后,判断是否到60秒,若不到则秒加一,然后返回,若到,则秒赋值为0,分加一,依次类推。包括日期显示的功能也是如此。另外,由于要实现倒计时功能,因此在中断程序中还要加入减一的寄存器,需要时将其进行显示。基于以上考虑,以R3为倒计时中的秒,R4为倒计时的分,当秒加1时R3减一,减到0之后,秒赋值为59,分减一,直到分为0。
流程框图如下:
图5.3.1 中断程序思路
6.1 软件介绍
在这部分工作中用到了keil C和Proteus两个软件,其中Keil C用来编译程序并生成hex文件,装入Proteus仿真图的芯片中,通过仿真结果一步步进行调整最后达到预期的功能。
9 电子钟设计中遇到的问题及其解决方法
在电子钟设计中程序比较长,遇到了各种各样的问题,比如跳转距离过长出现语法错误。因此修改程序的时间很长,有语法错误,也有仿真时功能无法实现的问题,在此不能一一说明,只能对个别问题加以阐述。
由于在走时调整、闹铃调整及定时设定时,按下键时每0.2秒步进1,这就需要每0.2秒对按键输入进行一次扫描,因此开始时,我用的是延迟0.2秒再返回按键扫描程序的方法,但是可以想到,进行单纯的延迟时,不会有显示,在仿真时功能没有实现。后来我采用的是反复调用若干次显示程序以实现延迟,较好地解决了这个问题。虽然现在回过头来发现这个问题很简单,但当时是百思不得其解。我从中学到:任何简单的事情都会有出错的可能,有些时候人的思路中的一些错误是自己难以发现的,在工作和生活中要多对自己的想法提出质疑。
不光是这种明显的问题很多,有些细微的问题也一样出现了,但解决起来并不容易。
在最后的硬件实现阶段发现蜂鸣器不响,此时连接方法是蜂鸣器直接连接在了P0.7引脚。考虑到单片机的输出电流比较小,虽然P0.7引脚的电平已经达到2V以上,也无法驱动蜂鸣器。于是我设想加上驱动电路。
作此修改之后,蜂鸣器发出响声,但是又遇到了新的问题:蜂鸣器无法停止响声!看上去这个问题很简单,只要有停止闹铃的按键输入时,在程序中加一句将P0.7置零的语句就可以了。但事实并不是这么简单。通过苦思冥想,发现在主程序中,对“定时时间是否已到”是不断进行扫描的,而倒计时是在中断服务子程序中进行的,当倒计时为0分0秒时,主程序判断出来后会不停地调用声音程序:
结论
本文介绍的是利用AT89C51单片机设计的有调时、定时、闹铃功能的电子钟。并详细说明了软件和硬件设计方法及仿真、硬件实现。在设计过程中可以看出,汇编语言有着其独特的魅力,它简单易学,语法错误容易纠正;用单片机实现电子钟的设计是比较方便和易于实现的。
通过电子钟的设计和制作,加深了对单片机的理解,能够更熟练地应用单片机实现预期的功能,对今后的工作有很大的帮助。电子钟各项功能的实现,为自动控制的实现打下了理论基础,获得了实现方法。
当然,该电子钟还有很多不足之处,比如闹钟不能关闭,且只能定一个闹钟。在今后的工作中还要对此加以研究,尽量实现更多的功能。
在今后的其他工作中,也可以把这次设计中的收获运用进去,这是我此次毕业设计得到的最大财富。
1 潘永雄.新编单片机原理与应用. 第2版.西安:西安电子科技大学出版社。
2 何立民.我国单片机应用技术发展趋势及展望.自动化与仪表。
3 向继文,廖立新.基于AT89S51 的电子钟系统设计.机电产品开发与创新,
4 张静.基于单片机数字钟的设计.办公自动化杂志。
5 樊金荣,谢智文.数字倒计时秒表的设计与实现.中南民族大学计算机科学学院院报。
附录:
数字钟原理图
PCB面板
原件清单:
Proteus设计图
Proteus仿真图形
程序
#include
//#include"DS18B20_3.H"
#include
#include
#define uint unsigned int
#define uchar unsigned char
#define wd 1 //定义是否有温度功能 =0时无温度,=1时有温度
#define yh 0x80 //LCD第一行的初始位置,因为LCD1602字符地址首位D7恒定为1(100000000=80)
#define er 0x80+0x40 //LCD第二行初始位置(因为第二行第一个字符位置地址是0x40)
//液晶屏的与C51之间的引脚连接定义(显示数据线接C51的P0口)
sbit en=P2^7;
sbit rw=P2^6; //如果硬件上rw接地,就不用写这句和后面的rw=0了
sbit rs=P2^5;
//校时按键与C51的引脚连接定义
sbit set=P3^0; //设置键
sbit add=P3^1; //加键
sbit dec=P3^2; //减键
sbit seeNL_NZ=P3^3; //查看农历/闹钟
sbit DQ=P3^7; //
sbit buzzer=P2^0; //蜂鸣器,通过三极管8550驱动,端口低电平响
sbit led=P2^4; //LCD背光开关
//DS1302时钟芯片与C51之间的引脚连接定义
sbit IO=P1^1;
sbit SCLK=P1^0;
sbit RST=P1^2;
uchar a,miao,shi,fen,ri,yue,nian,week,setn,temp;
uint flag;
//flag用于读取头文件中的温度值,和显示温度值
bit c_moon;
uchar nz_shi=12,nz_fen=0,nz_miao=0,setNZn; //定义闹钟变量
uchar shangyimiao,bsn,temp_hour; //记录上一秒时间
uchar T_NL_NZ; //计数器
bit timerOn=0; //闹钟启用标志位
bit baoshi=0; //整点报时标志位
bit p_r=0; //平年/润年 =0表示平年,=1表示润年
data uchar year_moon,month_moon,day_moon,week;
sbit ACC0=ACC^0;
sbit ACC7=ACC^7;
/************************************************************
ACC累加器=A
ACC.0=E0H
ACC.0就是ACC的第0位。Acc可以位寻址。
累加器ACC是一个8位的存储单元,是用来放数据的。但是,这个存储单元有其特殊的地位,
是单片机中一个非常关键的单元,很多运算都要通过ACC来进行。以后在学习指令时,
常用A来表示累加器。但有一些地方例外,比如在PUSH指令中,就必须用ACC这样的名字。
一般的说法,A代表了累加器中的内容、而ACC代表的是累加器的地址。
***************************************************************/
//********阳历转换阴历表************************************
code uchar year_code[597]={
0x04,0xAe,0x53, //1901 0
0x0A,0x57,0x48, //1902 3
0x55,0x26,0xBd, //1903 6
0x0d,0x26,0x50, //1904 9
0x0d,0x95,0x44, //1905 12
0x46,0xAA,0xB9, //1906 15
0x05,0x6A,0x4d, //1907 18
0x09,0xAd,0x42, //1908 21
0x24,0xAe,0xB6, //1909
0x04,0xAe,0x4A, //1910
0x6A,0x4d,0xBe, //1911
0x0A,0x4d,0x52, //1912
0x0d,0x25,0x46, //1913
0x5d,0x52,0xBA, //1914
0x0B,0x54,0x4e, //1915
0x0d,0x6A,0x43, //1916
0x29,0x6d,0x37, //1917
0x09,0x5B,0x4B, //1918
0x74,0x9B,0xC1, //1919
0x04,0x97,0x54, //1920
0x0A,0x4B,0x48, //1921
0x5B,0x25,0xBC, //1922
0x06,0xA5,0x50, //1923
0x06,0xd4,0x45, //1924
0x4A,0xdA,0xB8, //1925
0x02,0xB6,0x4d, //1926
0x09,0x57,0x42, //1927
0x24,0x97,0xB7, //1928
0x04,0x97,0x4A, //1929
0x66,0x4B,0x3e, //1930
0x0d,0x4A,0x51, //1931
0x0e,0xA5,0x46, //1932
0x56,0xd4,0xBA, //1933
0x05,0xAd,0x4e, //1934
0x02,0xB6,0x44, //1935
0x39,0x37,0x38, //1936
0x09,0x2e,0x4B, //1937
0x7C,0x96,0xBf, //1938
0x0C,0x95,0x53, //1939
0x0d,0x4A,0x48, //1940
0x6d,0xA5,0x3B, //1941
0x0B,0x55,0x4f, //1942
0x05,0x6A,0x45, //1943
0x4A,0xAd,0xB9, //1944
0x02,0x5d,0x4d, //1945
0x09,0x2d,0x42, //1946
0x2C,0x95,0xB6, //1947
0x0A,0x95,0x4A, //1948
0x7B,0x4A,0xBd, //1949
0x06,0xCA,0x51, //1950
0x0B,0x55,0x46, //1951
0x55,0x5A,0xBB, //1952
0x04,0xdA,0x4e, //1953
0x0A,0x5B,0x43, //1954
0x35,0x2B,0xB8, //1955
0x05,0x2B,0x4C, //1956
0x8A,0x95,0x3f, //1957
0x0e,0x95,0x52, //1958
0x06,0xAA,0x48, //1959
0x7A,0xd5,0x3C, //1960
0x0A,0xB5,0x4f, //1961
0x04,0xB6,0x45, //1962
0x4A,0x57,0x39, //1963
0x0A,0x57,0x4d, //1964
0x05,0x26,0x42, //1965
0x3e,0x93,0x35, //1966
0x0d,0x95,0x49, //1967
0x75,0xAA,0xBe, //1968
0x05,0x6A,0x51, //1969
0x09,0x6d,0x46, //1970
0x54,0xAe,0xBB, //1971
0x04,0xAd,0x4f, //1972
0x0A,0x4d,0x43, //1973
0x4d,0x26,0xB7, //1974
0x0d,0x25,0x4B, //1975
0x8d,0x52,0xBf, //1976
0x0B,0x54,0x52, //1977
0x0B,0x6A,0x47, //1978
0x69,0x6d,0x3C, //1979
0x09,0x5B,0x50, //1980
0x04,0x9B,0x45, //1981
0x4A,0x4B,0xB9, //1982
0x0A,0x4B,0x4d, //1983
0xAB,0x25,0xC2, //1984
0x06,0xA5,0x54, //1985
0x06,0xd4,0x49, //1986
0x6A,0xdA,0x3d, //1987
0x0A,0xB6,0x51, //1988
0x09,0x37,0x46, //1989
0x54,0x97,0xBB, //1990
0x04,0x97,0x4f, //1991
0x06,0x4B,0x44, //1992
0x36,0xA5,0x37, //1993
0x0e,0xA5,0x4A, //1994
0x86,0xB2,0xBf, //1995
0x05,0xAC,0x53, //1996
0x0A,0xB6,0x47, //1997
0x59,0x36,0xBC, //1998
0x09,0x2e,0x50, //1999 294
0x0C,0x96,0x45, //2000 297
0x4d,0x4A,0xB8, //2001
0x0d,0x4A,0x4C, //2002
0x0d,0xA5,0x41, //2003
0x25,0xAA,0xB6, //2004
0x05,0x6A,0x49, //2005
0x7A,0xAd,0xBd, //2006
0x02,0x5d,0x52, //2007
0x09,0x2d,0x47, //2008
0x5C,0x95,0xBA, //2009
0x0A,0x95,0x4e, //2010
0x0B,0x4A,0x43, //2011
0x4B,0x55,0x37, //2012
0x0A,0xd5,0x4A, //2013
0x95,0x5A,0xBf, //2014
0x04,0xBA,0x53, //2015
0x0A,0x5B,0x48, //2016
0x65,0x2B,0xBC, //2017
0x05,0x2B,0x50, //2018
0x0A,0x93,0x45, //2019
0x47,0x4A,0xB9, //2020
0x06,0xAA,0x4C, //2021
0x0A,0xd5,0x41, //2022
0x24,0xdA,0xB6, //2023
0x04,0xB6,0x4A, //2024
0x69,0x57,0x3d, //2025
0x0A,0x4e,0x51, //2026
0x0d,0x26,0x46, //2027
0x5e,0x93,0x3A, //2028
0x0d,0x53,0x4d, //2029
0x05,0xAA,0x43, //2030
0x36,0xB5,0x37, //2031
0x09,0x6d,0x4B, //2032
0xB4,0xAe,0xBf, //2033
0x04,0xAd,0x53, //2034
0x0A,0x4d,0x48, //2035
0x6d,0x25,0xBC, //2036
0x0d,0x25,0x4f, //2037
0x0d,0x52,0x44, //2038
0x5d,0xAA,0x38, //2039
0x0B,0x5A,0x4C, //2040
0x05,0x6d,0x41, //2041
0x24,0xAd,0xB6, //2042
0x04,0x9B,0x4A, //2043
0x7A,0x4B,0xBe, //2044
0x0A,0x4B,0x51, //2045
0x0A,0xA5,0x46, //2046
0x5B,0x52,0xBA, //2047
0x06,0xd2,0x4e, //2048
0x0A,0xdA,0x42, //2049
0x35,0x5B,0x37, //2050
0x09,0x37,0x4B, //2051
0x84,0x97,0xC1, //2052
0x04,0x97,0x53, //2053
0x06,0x4B,0x48, //2054
0x66,0xA5,0x3C, //2055
0x0e,0xA5,0x4f, //2056
0x06,0xB2,0x44, //2057
0x4A,0xB6,0x38, //2058
0x0A,0xAe,0x4C, //2059
0x09,0x2e,0x42, //2060
0x3C,0x97,0x35, //2061
0x0C,0x96,0x49, //2062
0x7d,0x4A,0xBd, //2063
0x0d,0x4A,0x51, //2064
0x0d,0xA5,0x45, //2065
0x55,0xAA,0xBA, //2066
0x05,0x6A,0x4e, //2067
0x0A,0x6d,0x43, //2068
0x45,0x2e,0xB7, //2069
0x05,0x2d,0x4B, //2070
0x8A,0x95,0xBf, //2071
0x0A,0x95,0x53, //2072
0x0B,0x4A,0x47, //2073
0x6B,0x55,0x3B, //2074
0x0A,0xd5,0x4f, //2075
0x05,0x5A,0x45, //2076
0x4A,0x5d,0x38, //2077
0x0A,0x5B,0x4C, //2078
0x05,0x2B,0x42, //2079
0x3A,0x93,0xB6, //2080
0x06,0x93,0x49, //2081
0x77,0x29,0xBd, //2082
0x06,0xAA,0x51, //2083
0x0A,0xd5,0x46, //2084
0x54,0xdA,0xBA, //2085
0x04,0xB6,0x4e, //2086
0x0A,0x57,0x43, //2087
0x45,0x27,0x38, //2088
0x0d,0x26,0x4A, //2089
0x8e,0x93,0x3e, //2090
0x0d,0x52,0x52, //2091
0x0d,0xAA,0x47, //2092
0x66,0xB5,0x3B, //2093
0x05,0x6d,0x4f, //2094
0x04,0xAe,0x45, //2095
0x4A,0x4e,0xB9, //2096
0x0A,0x4d,0x4C, //2097
0x0d,0x15,0x41, //2098
0x2d,0x92,0xB5, //2099
};
///月份数据表
code uchar day_code1[9]={0x0,0x1f,0x3b,0x5a,0x78,0x97,0xb5,0xd4,0xf3};
code uint day_code2[3]={0x111,0x130,0x14e};
/*
函数功能:输入BCD阳历数据,输出BCD阴历数据(只允许1901-20##年)
调用函数示例:Conversion(c_sun,year_sun,month_sun,day_sun)
如:计算20##年10月16日Conversion(0,0x4,0x10,0x16);
c_sun,year_sun,month_sun,day_sun均为BCD数据,c_sun为世纪标志位,c_sun=0为21世
纪,c_sun=1为19世纪
调用函数后,原有数据不变,读c_moon,year_moon,month_moon,day_moon得出阴历BCD数据
*/
bit c_moon;
data uchar year_moon,month_moon,day_moon,week;
//子函数,用于读取数据表中农历月的大月或小月,如果该月为大返回1,为小返回0
bit get_moon_day(uchar month_p,uint table_addr)
{
uchar temp;
switch (month_p){
case 1:{temp=year_code[table_addr]&0x08;
if (temp==0)return(0);else return(1);}
case 2:{temp=year_code[table_addr]&0x04;
if (temp==0)return(0);else return(1);}
case 3:{temp=year_code[table_addr]&0x02;
if (temp==0)return(0);else return(1);}
case 4:{temp=year_code[table_addr]&0x01;
if (temp==0)return(0);else return(1);}
case 5:{temp=year_code[table_addr+1]&0x80;
if (temp==0) return(0);else return(1);}
case 6:{temp=year_code[table_addr+1]&0x40;
if (temp==0)return(0);else return(1);}
case 7:{temp=year_code[table_addr+1]&0x20;
if (temp==0)return(0);else return(1);}
case 8:{temp=year_code[table_addr+1]&0x10;
if (temp==0)return(0);else return(1);}
case 9:{temp=year_code[table_addr+1]&0x08;
if (temp==0)return(0);else return(1);}
case 10:{temp=year_code[table_addr+1]&0x04;
if (temp==0)return(0);else return(1);}
case 11:{temp=year_code[table_addr+1]&0x02;
if (temp==0)return(0);else return(1);}
case 12:{temp=year_code[table_addr+1]&0x01;
if (temp==0)return(0);else return(1);}
case 13:{temp=year_code[table_addr+2]&0x80;
if (temp==0)return(0);else return(1);}
}
}
/*
函数功能:输入BCD阳历数据,输出BCD阴历数据(只允许1901-20##年)
调用函数示例:Conversion(c_sun,year_sun,month_sun,day_sun)
如:计算20##年10月16日Conversion(0,0x4,0x10,0x16);
c_sun,year_sun,month_sun,day_sun均为BCD数据,c_sun为世纪标志位,c_sun=0为21世
纪,c_sun=1为19世纪
调用函数后,原有数据不变,读c_moon,year_moon,month_moon,day_moon得出阴历BCD数据
*/
void Conversion(bit c,uchar year,uchar month,uchar day)
{ //c=0 为21世纪,c=1 为19世纪 输入输出数据均为BCD数据
uchar temp1,temp2,temp3,month_p;
uint temp4,table_addr;
bit flag2,flag_y;
temp1=year/16; //BCD->hex 先把数据转换为十六进制
temp2=year%16;
// year=temp1*10+temp2;
year=temp1*16+temp2;
temp1=month/16;
temp2=month%16;
//month=temp1*10+temp2;
month=temp1*16+temp2;
temp1=day/16;
temp2=day%16;
//day=temp1*10+temp2;
day=temp1*16+temp2;
//定位数据表地址
if(c==0){
table_addr=(year+0x64-1)*0x3;
}
else {
table_addr=(year-1)*0x3;
}
//定位数据表地址完成
//取当年春节所在的公历月份
temp1=year_code[table_addr+2]&0x60;
temp1=_cror_(temp1,5);
//取当年春节所在的公历月份完成
//取当年春节所在的公历日
temp2=year_code[table_addr+2]&0x1f;
//取当年春节所在的公历日完成
// 计算当年春年离当年元旦的天数,春节只会在公历1月或2月
if(temp1==0x1){
temp3=temp2-1;
}
else{
temp3=temp2+0x1f-1;
}
// 计算当年春年离当年元旦的天数完成
//计算公历日离当年元旦的天数,为了减少运算,用了两个表
//day_code1[9],day_code2[3]
//如果公历月在九月或前,天数会少于0xff,用表day_code1[9],
//在九月后,天数大于0xff,用表day_code2[3]
//如输入公历日为8月10日,则公历日离元旦天数为day_code1[8-1]+10-1
//如输入公历日为11月10日,则公历日离元旦天数为day_code2[11-10]+10-1
if (month<10){
temp4=day_code1[month-1]+day-1;
}
else{
temp4=day_code2[month-10]+day-1;
}
if ((month>0x2)&&(year%0x4==0)){ //如果公历月大于2月并且该年的2月为闰月,天数加1
temp4+=1;
}
//计算公历日离当年元旦的天数完成
//判断公历日在春节前还是春节后
if (temp4>=temp3){ //公历日在春节后或就是春节当日使用下面代码进行运算
temp4-=temp3;
month=0x1;
month_p=0x1; //month_p为月份指向,公历日在春节前或就是春节当日month_p指向首月
flag2=get_moon_day(month_p,table_addr); //检查该农历月为大小还是小月,大月返回1,小月返回0
flag_y=0;
if(flag2==0)temp1=0x1d; //小月29天
else temp1=0x1e; //大小30天
temp2=year_code[table_addr]&0xf0;
temp2=_cror_(temp2,4); //从数据表中取该年的闰月月份,如为0则该年无闰月
while(temp4>=temp1){
temp4-=temp1;
month_p+=1;
if(month==temp2){
flag_y=~flag_y;
if(flag_y==0)month+=1;
}
else month+=1;
flag2=get_moon_day(month_p,table_addr);
if(flag2==0)temp1=0x1d;
else temp1=0x1e;
}
day=temp4+1;
}
else{ //公历日在春节前使用下面代码进行运算
temp3-=temp4;
if (year==0x0){year=0x63;c=1;}
else year-=1;
table_addr-=0x3;
month=0xc;
temp2=year_code[table_addr]&0xf0;
temp2=_cror_(temp2,4);
if (temp2==0)
month_p=0xc;
else
month_p=0xd; //
//month_p为月份指向,如果当年有闰月,一年有十三个月,月指向13,无闰月指向12
flag_y=0;
flag2=get_moon_day(month_p,table_addr);
if(flag2==0)temp1=0x1d;
else temp1=0x1e;
while(temp3>temp1){
temp3-=temp1;
month_p-=1;
if(flag_y==0)month-=1;
if(month==temp2)flag_y=~flag_y;
flag2=get_moon_day(month_p,table_addr);
if(flag2==0)temp1=0x1d;
else temp1=0x1e;
}
day=temp1-temp3+1;
}
c_moon=c; //HEX->BCD ,运算结束后,把数据转换为BCD数据
temp1=year/10;
temp1=_crol_(temp1,4);
temp2=year%10;
year_moon=temp1|temp2;
temp1=month/10;
temp1=_crol_(temp1,4);
temp2=month%10;
month_moon=temp1|temp2;
temp1=day/10;
temp1=_crol_(temp1,4);
temp2=day%10;
day_moon=temp1|temp2;
}
/*
函数功能:输入BCD阳历数据,输出BCD星期数据(只允许1901-20##年)
调用函数示例:Conver_week(c_sun,year_sun,month_sun,day_sun)
如:计算20##年10月16日Conversion(0,0x4,0x10,0x16);
c_sun,year_sun,month_sun,day_sun均为BCD数据,c_sun为世纪标志位,c_sun=0为21世
纪,c_sun=1为19世纪
调用函数后,原有数据不变,读week得出阴历BCD数据
*/
code uchar table_week[12]={0,3,3,6,1,4,6,2,5,0,3,5}; //月修正数据表
/*
算法:日期+年份+所过闰年数+月较正数之和除7 的余数就是星期但如果是在
闰年又不到3 月份上述之和要减一天再除7
星期数为0
*/
void Conver_week(uchar year,uchar month,uchar day)
{//c=0 为21世纪,c=1 为19世纪 输入输出数据均为BCD数据
uchar p1,p2;
year+=0x64; //如果为21世纪,年份数加100
p1=year/0x4; //所过闰年数只算1900年之后的
p2=year+p1;
p2=p2%0x7; //为节省资源,先进行一次取余,避免数大于0xff,避免使用整型数据
p2=p2+day+table_week[month-1];
if (year%0x4==0&&month<3)p2-=1;
week=p2%0x7;
}
/**************************************************************/
uchar code tab1[]={"20 - - "}; //年显示的固定字符
uchar code tab2[]={" : : "}; //时间显示的固定字符
uchar code nlp[]={"NL: - - PING"}; //农历平年显示
uchar code nlr[]={"NL: - - RUN "}; //农历润年显示
uchar code NZd[]={"timer: - - "}; //显示闹钟固定点
uchar code qk[]= {" "}; //清空显示
uchar code tm[]= {"time"};
//=====================DS18B20=============================================
//******************************************************************************
void Delayns(int num)//延时函数
{
while(num--);
}
//******************************************************************************
void Init_DS18B20(void)//初始化ds1820
{
unsigned char x=0;
DQ = 1; //DQ复位
Delayns(8); //稍做延时
DQ = 0; //单片机将DQ拉低
Delayns(80); //精确延时 大于 480us
DQ = 1; //拉高总线
Delayns(14);
x=DQ; //稍做延时后 如果x=0则初始化成功 x=1则初始化失败
Delayns(20);
}
//******************************************************************************
unsigned char ReadOneChar(void)//读一个字节
{
unsigned char i=0;
unsigned char dat = 0;
for (i=8;i>0;i--)
{
DQ = 0; // 给脉冲信号
dat>>=1;
DQ = 1; // 给脉冲信号
if(DQ)
dat|=0x80;
Delayns(4);
}
return(dat);
}
//******************************************************************************
void WriteOneChar(unsigned char dat)//写一个字节
{
unsigned char i=0;
for (i=8; i>0; i--)
{
DQ = 0;
DQ = dat&0x01;
Delayns(5);
DQ = 1;
dat>>=1;
}
}
//******************************************************************************
unsigned int ReadTemperature(void)//读取温度
{
unsigned char a=0;
unsigned char b=0;
unsigned int t=0;
float tt=0;
Init_DS18B20();
WriteOneChar(0xCC); // 跳过读序号列号的操作
WriteOneChar(0x44); // 启动温度转换
Init_DS18B20();
WriteOneChar(0xCC); //跳过读序号列号的操作
WriteOneChar(0xBE); //读取温度寄存器
a=ReadOneChar(); //读低8位
b=ReadOneChar(); //读高8位
t=b;
t<<=8;
t=t|a;
tt=t*0.0625;
t= tt*10+0.5; //放大10倍输出并四舍五入
return(t);
}
//*****************************************************************************
//延时函数,后面经常调用
void delay(uint xms)//延时函数,有参函数
{
uint x,y;
for(x=xms;x>0;x--)
for(y=110;y>0;y--);
}
/********液晶写入指令函数与写入数据函数,以后可调用**************/
/*在这个程序中,液晶写入有关函数会在DS1302的函数中调用,所以液晶程序要放在前面*/
write_1602com(uchar com)//****液晶写入指令函数****
{
rs=0;//数据/指令选择置为指令
rw=0; //读写选择置为写
P0=com;//送入数据
delay(1);
en=1;//拉高使能端,为制造有效的下降沿做准备
delay(1);
en=0;//en由高变低,产生下降沿,液晶执行命令
}
write_1602dat(uchar dat)//***液晶写入数据函数****
{
rs=1;//数据/指令选择置为数据
rw=0; //读写选择置为写
P0=dat;//送入数据
delay(1);
en=1; //en置高电平,为制造下降沿做准备
delay(1);
en=0; //en由高变低,产生下降沿,液晶执行命令
}
lcd_init()//***液晶初始化函数****
{
write_1602com(0x38);//设置液晶工作模式,意思:16*2行显示,5*7点阵,8位数据
write_1602com(0x0c);//开显示不显示光标
write_1602com(0x06);//整屏不移动,光标自动右移
write_1602com(0x01);//清显示
write_1602com(yh+1);//日历显示固定符号从第一行第1个位置之后开始显示
for(a=0;a<14;a++)
{
write_1602dat(tab1[a]);//向液晶屏写日历显示的固定符号部分
//delay(3);
}
write_1602com(er);//时间显示固定符号写入位置,从第2个位置后开始显示
for(a=0;a<8;a++)
{
write_1602dat(tab2[a]);//写显示时间固定符号,两个冒号
//delay(3);
}
}
/*********************over***********************/
/***************DS1302有关子函数********************/
void write_byte(uchar dat)//写一个字节
{
ACC=dat;
RST=1;
for(a=8;a>0;a--)
{
IO=ACC0;
SCLK=0;
SCLK=1;
ACC=ACC>>1;
}
}
uchar read_byte()//读一个字节
{
RST=1;
for(a=8;a>0;a--)
{
ACC7=IO;
SCLK=1;
SCLK=0;
ACC=ACC>>1;
}
return (ACC);
}
//----------------------------------------
void write_1302(uchar add,uchar dat)//向1302芯片写函数,指定写入地址,数据
{
RST=0;
SCLK=0;
RST=1;
write_byte(add);
write_byte(dat);
SCLK=1;
RST=0;
}
uchar read_1302(uchar add)//从1302读数据函数,指定读取数据来源地址
{
uchar temp;
RST=0;
SCLK=0;
RST=1;
write_byte(add);
temp=read_byte();
SCLK=1;
RST=0;
return(temp);
}
uchar BCD_Decimal(uchar bcd)//BCD码转十进制函数,输入BCD,返回十进制
{
uchar Decimal;
Decimal=bcd>>4;
return(Decimal=Decimal*10+(bcd&=0x0F));
}
//--------------------------------------
void ds1302_init() //1302芯片初始化子函数(20##-01-07,12:00:00,week4)
{
RST=0;
SCLK=0;
write_1302(0x8e,0x00); //允许写,禁止写保护
//write_1302(0x80,0x00); //向DS1302内写秒寄存器80H写入初始秒数据00
//write_1302(0x82,0x00);//向DS1302内写分寄存器82H写入初始分数据00
//write_1302(0x84,0x12);//向DS1302内写小时寄存器84H写入初始小时数据12
//write_1302(0x8a,0x04);//向DS1302内写周寄存器8aH写入初始周数据4
//write_1302(0x86,0x07);//向DS1302内写日期寄存器86H写入初始日期数据07
//write_1302(0x88,0x01);//向DS1302内写月份寄存器88H写入初始月份数据01
//write_1302(0x8c,0x10);//向DS1302内写年份寄存器8cH写入初始年份数据10
write_1302(0x8e,0x80); //打开写保护
}
//------------------------------------
//温度显示子函数
void write_temp(uchar add,uint dat)//向LCD写温度数据,并指定显示位置
{
uint gw,sw,bw;
bw=dat/100;//取得百位
sw=dat%100/10;//取得十位数字
gw=dat%10;//取得个位数字
write_1602com(er+add);//er是头文件规定的值0x80+0x40
write_1602dat(0x30+bw);
write_1602dat(0x30+sw);//数字+30得到该数字的LCD1602显示码
write_1602dat('.');
write_1602dat(0x30+gw);//数字+30得到该数字的LCD1602显示码
write_1602dat(0xdf);//显示温度的小圆圈符号,0xdf是液晶屏字符库的该符号地址码
write_1602dat(0x43); //显示"C"符号,0x43是液晶屏字符库里大写C的地址码
}
//------------------------------------
//时分秒显示子函数
void write_sfm(uchar add,uchar dat)//向LCD写时分秒,有显示位置加、现示数据,两个参数
{
uchar gw,sw;
gw=dat%10;//取得个位数字
sw=dat/10;//取得十位数字
write_1602com(er+add);//er是头文件规定的值0x80+0x40
write_1602dat(0x30+sw);//数字+30得到该数字的LCD1602显示码
write_1602dat(0x30+gw);//数字+30得到该数字的LCD1602显示码
}
//-------------------------------------
//年月日显示子函数
void write_nyr(uchar add,uchar dat)//向LCD写年月日,有显示位置加数、显示数据,两个参数
{
uchar gw,sw;
gw=dat%10;//取得个位数字
sw=dat/10;//取得十位数字
write_1602com(yh+add);//设定显示位置为第一个位置+add
write_1602dat(0x30+sw);//数字+30得到该数字的LCD1602显示码
write_1602dat(0x30+gw);//数字+30得到该数字的LCD1602显示码
}
//------------------------------------
//农历显示子函数
void write_nl(uchar add,uchar dat)//向LCD写时分秒,有显示位置加、现示数据,两个参数
{
uchar gw,sw;
//gw=dat%10;//取得个位数字
//sw=dat/10;//取得十位数字
gw=dat%16;//取得个位数字
sw=dat/16;//取得十位数字
write_1602com(er+add);//er是头文件规定的值0x80+0x40
// write_1602dat(0x30+sw);//数字+30得到该数字的LCD1602显示码
// write_1602dat(0x30+gw);//数字+30得到该数字的LCD1602显示码
write_1602dat('0'+sw);//数字+30得到该数字的LCD1602显示码
write_1602dat('0'+gw);//数字+30得到该数字的LCD1602显示码
}
//-------------------------------------------
void write_week(uchar week)//写星期函数
{
write_1602com(yh+0x0c);//星期字符的显示位置
switch(week)
{
case 1:write_1602dat('M');//星期数为1时,显示
write_1602dat('O');
write_1602dat('N');
break;
case 2:write_1602dat('T');//星期数据为2时显示
write_1602dat('U');
write_1602dat('E');
break;
case 3:write_1602dat('W');//星期数据为3时显示
write_1602dat('E');
write_1602dat('D');
break;
case 4:write_1602dat('T');//星期数据为4是显示
write_1602dat('H');
write_1602dat('U');
break;
case 5:write_1602dat('F');//星期数据为5时显示
write_1602dat('R');
write_1602dat('I');
break;
case 6:write_1602dat('S');//星期数据为6时显示
write_1602dat('T');
write_1602dat('A');
break;
case 0:write_1602dat('S');//星期数据为7时显示
write_1602dat('U');
write_1602dat('N');
break;
}
}
//****************键盘扫描有关函数**********************
void keyscan()
{
if(seeNL_NZ==0)
{
delay(9);
if(seeNL_NZ==0)
{
if((setn==0)&&(setNZn==0)) //在没有进入调时模式时才可按动
{
buzzer=0;//蜂鸣器短响一次
delay(20);
buzzer=1;
if(TR1==1)
{
TR1=0;
}
else
{
T_NL_NZ++;
if(T_NL_NZ==3)
{
setn=0;
setNZn=0;
T_NL_NZ=0;
}
}
}
while(seeNL_NZ==0);
}
}
if(set==0)//---------------set为功能键(设置键)--------------------
{
delay(9);//延时,用于消抖动
if(set==0)//延时后再次确认按键按下
{
buzzer=0;//蜂鸣器短响一次
delay(20);
buzzer=1;
while(!set);
if(T_NL_NZ==0x02) //证明是对闹钟进行设置
{
setNZn++;
if(setNZn==4) //闹钟设定成功,退回到正常显示并开启闹钟
{
setNZn=0;
setn=0;
timerOn=1;
}
switch(setNZn)
{
case 0: //正常显示日期时间
write_1602com(0x0c); //?柚霉獗瓴簧了?
write_1602com(er); //时间显示固定符号写入位置?
for(a=0;a<16;a++)
write_1602dat(NZd[a]); //写显示时间固定符号,两个冒号
write_sfm(8,nz_shi); //闹钟 时
write_sfm(11,nz_fen); //闹钟 分
write_sfm(14,nz_miao); //闹钟 秒
break;
case 1: //闹钟秒光标闪烁
write_1602com(er+15); //设置按键按动一次,秒位置显示光标 //er+0x09;
write_1602com(0x0f); //设置光标为闪烁
break;
case 2: //闹钟分光标闪烁
write_1602com(er+12); //设置按键按动一次,秒位置显示光标 //er+0x09;
write_1602com(0x0f); //设置光标为闪烁
break;
case 3: //闹钟时光标闪烁
write_1602com(er+9); //设置按键按动一次,秒位置显示光标 //er+0x09;
write_1602com(0x0f); //设置光标为闪烁
break;
}
}
else //证明是对时间及日期进行设置
{
if(T_NL_NZ==0)
{
setn++;
if(setn==7)
setn=0; //设置按键共有秒、分、时、星期、日、月、年、返回,8个功能循环
switch(setn)
{
case 1: TR0=0;//关闭定时器
//TR1=0;
write_1602com(er+7);//设置按键按动一次,秒位置显示光标 //er+0x09;
write_1602com(0x0f);//设置光标为闪烁
temp=(miao)/10*16+(miao)%10;//秒数据写入DS1302
write_1302(0x8e,0x00);
write_1302(0x80,0x80|temp);//miao
write_1302(0x8e,0x80);
break;
case 2:
write_1602com(er+4); //按2次fen位置显示光标 //er+0x06
//write_1602com(0x0f);
break;
case 3:
write_1602com(er+1); //按动3次,shi
//write_1602com(0x0f);
break;
// case 4: write_1602com(yh+0x0e);//按动4次,week
//write_1602com(0x0f);
// break;
case 4: write_1602com(yh+0x0a);//按动4次,ri
//write_1602com(0x0f);
break;
case 5: write_1602com(yh+0x07);//按动5次,yue
//write_1602com(0x0f);
break;
case 6: write_1602com(yh+0x04);//按动6次,nian
//write_1602com(0x0f);
break;
case 0:
write_1602com(0x0c);//按动到第7次,设置光标不闪烁
TR0=1;//打开定时器
temp=(miao)/10*16+(miao)%10;
write_1302(0x8e,0x00);
write_1302(0x80,0x00|temp);//miao数据写入DS1302
write_1302(0x8e,0x80);
break;
}
}
}
}
}
//------------------------------加键add----------------------------
if((setn!=0)&&(setNZn==0))//当set按下以下。再按以下键才有效(按键次数不等于零)
{
if(add==0) //上调键
{
delay(10);
if(add==0)
{
buzzer=0;//蜂鸣器短响一次
delay(20);
buzzer=1;
while(!add);
switch(setn)
{
case 1:miao++;//设置键按动1次,调秒
if(miao==60)
miao=0;//秒超过59,再加1,就归零
write_sfm(0x06,miao);//令LCD在正确位置显示"加"设定好的秒数
temp=(miao)/10*16+(miao)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00); //允许写,禁止写保护
write_1302(0x80,temp); //向DS1302内写秒寄存器80H写入调整后的秒数据BCD码
write_1302(0x8e,0x80); //打开写保护
write_1602com(er+7);//因为设置液晶的模式是写入数据后,光标自动右移,所以要指定返回
//write_1602com(0x0b);
break;
case 2:fen++;
if(fen==60)
fen=0;
write_sfm(0x03,fen);//令LCD在正确位置显示"加"设定好的分数据
temp=(fen)/10*16+(fen)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x82,temp);//向DS1302内写分寄存器82H写入调整后的分数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(er+4);//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置
break;
case 3:shi++;
if(shi==24)
shi=0;
write_sfm(0x00,shi);//令LCD在正确的位置显示"加"设定好的小时数据
temp=(shi)/10*16+(shi)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x84,temp);//向DS1302内写小时寄存器84H写入调整后的小时数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(er+1);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
/*
case 4:week++;
if(week==8)
week=1;
write_1602com(yh+0x0C);//指定'加'后的周数据显示位置
write_week(week);//指定周数据显示内容
temp=(week)/10*16+(week)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x8a,temp);//向DS1302内写周寄存器8aH写入调整后的周数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+0x0e);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
*/
case 4:ri++;
if(ri==32)
ri=1;
Conver_week(nian,yue,ri);
write_week(week);
write_nyr(9,ri);//令LCD在正确的位置显示"加"设定好的日期数据
temp=(ri)/10*16+(ri)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x86,temp);//向DS1302内写日期寄存器86H写入调整后的日期数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+10);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
case 5:yue++;
if(yue==13)
yue=1;
Conver_week(nian,yue,ri);
write_week(week);
write_nyr(6,yue);//令LCD在正确的位置显示"加"设定好的月份数据
temp=(yue)/10*16+(yue)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x88,temp);//向DS1302内写月份寄存器88H写入调整后的月份数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+7);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
case 6:nian++;
if(nian==100)
nian=0;
Conver_week(nian,yue,ri);
write_week(week);
write_nyr(3,nian);//令LCD在正确的位置显示"加"设定好的年份数据
temp=(nian)/10*16+(nian)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x8c,temp);//向DS1302内写年份寄存器8cH写入调整后的年份数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+4);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
}
}
}
//------------------减键dec,各句功能参照'加键'注释---------------
if(dec==0)
{
delay(10);//调延时,消抖动
if(dec==0)
{
buzzer=0;//蜂鸣器短响一次
delay(20);
buzzer=1;
while(!dec);
switch(setn)
{
case 1:
miao--;
if(miao==-1)
miao=59;//秒数据减到-1时自动变成59
write_sfm(0x06,miao);//在LCD的正确位置显示改变后新的秒数
temp=(miao)/10*16+(miao)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00); //允许写,禁止写保护
write_1302(0x80,temp); //向DS1302内写秒寄存器80H写入调整后的秒数据BCD码
write_1302(0x8e,0x80); //打开写保护
write_1602com(er+7);//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置
//write_1602com(0x0b);
break;
case 2:
fen--;
if(fen==-1)
fen=59;
write_sfm(3,fen);
temp=(fen)/10*16+(fen)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x82,temp);//向DS1302内写分寄存器82H写入调整后的分数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(er+4);//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置
break;
case 3:
shi--;
if(shi==-1)
shi=23;
write_sfm(0,shi);
temp=(shi)/10*16+(shi)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x84,temp);//向DS1302内写小时寄存器84H写入调整后的小时数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(er+1);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
/*
case 4:week--;
if(week==0)
week=7;
write_1602com(yh+0x0C);//指定'加'后的周数据显示位置
write_week(week);//指定周数据显示内容
temp=(week)/10*16+(week)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x8a,temp);//向DS1302内写周寄存器8aH写入调整后的周数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+0x0e);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
*/
case 4:
ri--;
if(ri==0)
ri=31;
Conver_week(nian,yue,ri);
write_week(week);
write_nyr(9,ri);
temp=(ri)/10*16+(ri)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x86,temp);//向DS1302内写日期寄存器86H写入调整后的日期数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+10);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
case 5:
yue--;
if(yue==0)
yue=12;
Conver_week(nian,yue,ri);
write_week(week);
write_nyr(6,yue);
temp=(yue)/10*16+(yue)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x88,temp);//向DS1302内写月份寄存器88H写入调整后的月份数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+7);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
case 6:
nian--;
if(nian==-1)
nian=99;
Conver_week(nian,yue,ri);
write_week(week);
write_nyr(3,nian);
temp=(nian)/10*16+(nian)%10;//十进制转换成DS1302要求的DCB码
write_1302(0x8e,0x00);//允许写,禁止写保护
write_1302(0x8c,temp);//向DS1302内写年份寄存器8cH写入调整后的年份数据BCD码
write_1302(0x8e,0x80);//打开写保护
write_1602com(yh+4);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
}
}
}
}
if((setNZn!=0)&&(setn==0))
{
if(add==0) //上调键
{
delay(10);
if(add==0)
{
buzzer=0;//蜂鸣器短响一次
delay(20);
buzzer=1;
while(!add);
switch(setNZn)
{
case 1:
nz_miao++; //设置键按动1次,调秒
if(nz_miao==60)
nz_miao=0;//秒超过59,再加1,就归零
write_sfm(14,nz_miao);//令LCD在正确位置显示"加"设定好的秒数
write_1602com(er+15);//因为设置液晶的模式是写入数据后,光标自动右移,所以要指定返回
break;
case 2:
nz_fen++;
if(nz_fen==60)
nz_fen=0;
write_sfm(11,nz_fen);//令LCD在正确位置显示"加"设定好的分数据
write_1602com(er+12);//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置
break;
case 3:
nz_shi++;
if(nz_shi==24)
nz_shi=0;
write_sfm(8,nz_shi);//令LCD在正确的位置显示"加"设定好的小时数据
write_1602com(er+9);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
}
}
}
//------------------减键dec,各句功能参照'加键'注释---------------
if(dec==0)
{
delay(10);//调延时,消抖动
if(dec==0)
{
buzzer=0;//蜂鸣器短响一次
delay(20);
buzzer=1;
while(!dec);
switch(setNZn)
{
case 1:
nz_miao--;
if(nz_miao==-1)
nz_miao=59;//秒数据减到-1时自动变成59
write_sfm(14,nz_miao);//在LCD的正确位置显示改变后新的秒数
write_1602com(er+15);
break;
case 2:
nz_fen--;
if(nz_fen==-1)
nz_fen=59;
write_sfm(11,nz_fen);
write_1602com(er+12);//因为设置液晶的模式是写入数据后,指针自动加一,在这里是写回原来的位置
break;
case 3:
nz_shi--;
if(nz_shi==-1)
nz_shi=23;
write_sfm(8,nz_shi);
write_1602com(er+9);//因为设置液晶的模式是写入数据后,指针自动加一,所以需要光标回位
break;
}
}
}
}
}
//-------------------------------
void init(void) //定时器、计数器设置函数
{
TMOD=0x11; //指定定时/计数器的工作方式为3
TH0=0; //定时器T0的高四位=0
TL0=0; //定时器T0的低四位=0
TH1=0x3C;
TL1=0xB0;
EA=1; //系统允许有开放的中断
ET0=1; //允许T0中断
ET1=1;
IT1=1;
IT0=0;
TR0=1; //开启中断,启动定时器
TR1=0;
}
void alarm(void)
{
if((shi==nz_shi)&&(fen==nz_fen)&&(miao==0))
{
TR1=1;
}
if((shi==nz_shi)&&(fen==(nz_fen+1)))
{
TR1=0;
buzzer=1;
}
}
void ZD_baoshi(void)
{
buzzer=0;
delay(5);
buzzer=1;
bsn++;
if(bsn==temp_hour)
{
baoshi=0;
}
}
//*******************主函数**************************
//***************************************************
void main()
{
P1=0xff;
lcd_init(); //调用液晶屏初始化子函数
ds1302_init(); //调用DS1302时钟的初始化子函数
init(); //调用定时计数器的设置子函数
led=0; //打开LCD的背光电源
buzzer=0; //蜂鸣器长响一次
delay(80);
buzzer=1;
while(1) //无限循环下面的语句:
{
keyscan(); //调用键盘扫描子函数
if(timerOn==1)
alarm(); //闹钟输出
if((fen==0)&&(miao==0))
{
if(shi>12)
temp_hour=shi-12;
else
{
if(shi==0)
temp_hour=12;
else
temp_hour=shi;
}
shangyimiao=miao;
baoshi=1;
}
if(baoshi==1)
{
ZD_baoshi();
do
keyscan();
while(shangyimiao==miao);
shangyimiao=miao;
}
}
}
void timer0() interrupt 1 //取得并显示日历和时间
{
//Init_DS18B20();//温度传感器DS18b2初始化子函数,在头文件中
// flag=ReadTemperature();//将18b2头文件运行返回的函数结果送到变量FLAG中,用于显示
//读取秒时分周日月年七个数据(DS1302的读寄存器与写寄存器不一样):
miao = BCD_Decimal(read_1302(0x81));
fen = BCD_Decimal(read_1302(0x83));
shi = BCD_Decimal(read_1302(0x85));
ri = BCD_Decimal(read_1302(0x87));
yue = BCD_Decimal(read_1302(0x89));
nian=BCD_Decimal(read_1302(0x8d));
//week=BCD_Decimal(read_1302(0x8b)); //不读取,直接通过日期计算得到
if(T_NL_NZ==1) //显示农历
{
uint nian_temp,temp;
temp=nian;
nian_temp=2000+(temp&0xF0)*10+temp&0x0F;
if((nian_temp%400==0)||((nian_temp%100!=0)&&(nian_temp%4==0))) //判断是否为闰年
p_r=1;
else
p_r=0;
Conversion(0,nian,yue,ri);
write_1602com(er);//时间显示固定符号写入位置?
for(a=0;a<16;a++)
{
if(p_r==0)
write_1602dat(nlp[a]);//写显示时间固定符号,两个冒号
else
write_1602dat(nlr[a]);
}
write_nl(3,year_moon);//农历 年
write_nl(6,month_moon);//农历 月
write_nl(9,day_moon);//农历 日
do
keyscan();
while(T_NL_NZ==1);
write_1602com(er);//时间显示固定符号写入位置,从第2个位置后开始显示
for(a=0;a<16;a++)
{
write_1602dat(qk[a]);//写显示时间固定符号,两个冒号
}
write_1602com(er);//时间显示固定符号写入位置,从第2个位置后开始显示
for(a=0;a<8;a++)
{
write_1602dat(tab2[a]);//写显示时间固定符号,两个冒号
}
}
if(T_NL_NZ==2) //显示闹钟时间,
{
write_1602com(er);//时间显示固定符号写入位置?
for(a=0;a<16;a++)
write_1602dat(NZd[a]);//写显示时间固定符号,两个冒号
write_sfm(8,nz_shi);//农历 年
write_sfm(11,nz_fen);//农历 月
write_sfm(14,nz_miao);//农历 日
do
keyscan();
while(T_NL_NZ==2);
write_1602com(er);//时间显示固定符号写入位置,从第2个位置后开始显示
for(a=0;a<16;a++)
{
write_1602dat(qk[a]);//写显示时间固定符号,两个冒号
}
write_1602com(er);//时间显示固定符号写入位置,从第2个位置后开始显示
for(a=0;a<8;a++)
{
write_1602dat(tab2[a]);//写显示时间固定符号,两个冒号
}
}
else
{
//显示温度、秒、时、分数据:
if(wd)
{
flag=ReadTemperature()-5;
write_temp(10,flag);//显示温度,从第二行第12个字符后开始显示
}
else
{
write_1602com(er+12);
for(a=0;a<4;a++)
{
write_1602dat(tm[a]);
}
}
write_sfm(6,miao);//秒,从第二行第8个字后开始显示(调用时分秒显示子函数)
write_sfm(3,fen);//分,从第二行第5个字符后开始显示
write_sfm(0,shi);//小时,从第二行第2个字符后开始显示
}
//显示日、月、年数据:
write_nyr(9,ri);//日期,从第二行第9个字符后开始显示
write_nyr(6,yue);//月份,从第二行第6个字符后开始显示
write_nyr(3,nian);//年,从第二行第3个字符后开始显示
Conver_week(nian,yue,ri);
write_week(week);
}
unsigned char count1;
void timer1() interrupt 3 //取得并显示日历和时间
{
TH1=0x3C;
TL1=0xB0;
TR1=1;
count1++;
if(count1==10)
{
count1=0;
buzzer=!buzzer;
}
}