单片机时钟程序
实验目的:通过
电子时钟的综合实验掌握单片机编程的基本思想,学会对一些周边芯片的综合利用。通过实验,真正理解所学知识。
实验要求:数码管型时钟,可以显示时间、日期、星期、闰年,并有整点报时及闹铃,农历功能。
实验器材:主要用到的芯片有:PCF8563时钟芯片,ZLG7290键盘检测及LED显示模块,蜂鸣器及51单片机。系统概述:先对时钟芯片初始化,然后取出数据,进行处理,农历转换,并判断是否整点及到达定时时间,根据判断到的按键,进行相应的处理及显示。硬件介绍:1、ZLG7290ZLG7290是一I2C接口键盘及LED驱动管理器件,提供数据译码和循环、移位、段寻址等控制。它可采样64个按键或传感器,单片即可完成LED显示、键盘接口的全部功能。ZLG7290的从地址为70H,器件内部通过I2C总线访问的寄存器地址范围为00H~17H,任一寄存器都可按字节直接读写,并支持自动增址功能和地址翻转功能。使用 ZLG7290 驱动数码管显示有两种
,第一种方法是向命令缓冲区(07H-08H)写入复合指令,向 07H 写入命令并选通相应的数码管,向 08H 写入所要显示的数据,这种方法每次只能写入一个字节的数据,多字节数据的输出可在程序中用循环写入的方法实现;第二种方法是向显示缓存寄存器(10H~17H)写入所要显示的数据的段码,段码的编码规则为从高位到低位为 abcdefgdp ,这种方法每次可写入 1~8 个字节数据。ZLG7290 读普通键的入口地址和读功能键的入口地址不同,读普通按键的地址为 01H,读功能键的地址为 03H。读普通键返回按键的编号,读功能键返回的不是按键编号,需要程序对返回值进行翻译,转换成功能键的编号。2:PCF8563时钟芯片PCF8563是PHILIPS公司生产的低功耗CMOS实时时钟/日历芯片,芯片最大总线速度为400kbits/s,每次读写数据后,其内嵌的字地址寄存器器会自动产生增量。PCF8563可广泛应用于移动电话、便携仪器、传真机、电池电源等产品中。
PCF8563有16个8位寄存器,其中包括:可自动增量的地址寄存器、内置32.768kHz的振荡器(带有一个内部集成电容)、分频器(用于给实时时钟RTC提供源时钟)、可编程时钟输出、定时器、报警器、掉电检测器和400kHz的I2C总线接口。
所有16个寄存器设计成可寻址的8位并行寄存器,但不是所有位都有用。当一个RTC寄存器被读时,所有计数器的内容将被锁存,因此,在传送条件下,可以禁止对时钟/日历芯片的错读。
3:I2C介绍
1、只要求两条总线线路:一条串行数据线SDA,一条串行时钟线SCL;
2、每个连接到总线的器件都可以通过唯一的地址和一直存在的简单的主机/从机关系软件设定地址,主机可以作为主机发送器或主机接收器;
3、它是一个真正的多主机总线,如果两个或更多主机同时初始化,数据传输可以通过冲突检测和仲裁防止数据被破坏;
4、串行的8 位双向数据传输位速率在
模式下可达100kbit/s,快速模式下可达400kbit/s,高速模式下可达3.4Mbit/s;
5、连接到相同总线的IC 数量只受到总线的最大电容400pF 限制。
实验原理:选用的时钟芯片及显示模块均采用了I2C总线标准,使整个系统接线更简单,并且数据传输更简单。
程序主要由以下几部分组成:
I2C读写: 写数据 ISendStr(uchar sla,uchar suba,uchar *s,uchar no) ;
读数据IRcvStr(uchar sla,uchar suba,uchar *s,uchar no);
ZLG7290数码管显示及键盘取键值:
发送数据ZLG7290_SendData(unsigned char SubAdd,unsigned char Data);
显示数据ZLG7290_SendBuf(unsigned char * disp_buf,unsigned char num);
发送命令ZLG7290_SendCmd(unsigned char Data1,unsigned char Data2);
取键值 ZLG7290_GetKey();
农历转换部分,闹铃显示,整点报时等。
程序开始,首先初始化PCF8563时钟芯片,然后进入while循环,进行显示及键盘判断。程
序设置了几个全局变量,专门用于存储按键次数,达到一键多用的效果。下面的程序根据按键的次数,选择处理不同的数据或进行不同的操作。
本程序共用了5个按键,其中1键专门用于切换显示,2键用于修改时间及星期,3
键用于调整日期,4键用于修改闹铃。5键专门用于调整。
c程序:
#include"reg52.h"
#include"VIIC_C51.H"
#include"zlg7290.h"
#include "intrins.h"
#include "18b20.h" //温度:ReadTemperature()
#define PCF8563 0xA2 //定义器件地址
#define WRADDR 0x00 //定义写单元的首地址
#define RDADDR 0x02 //定义读单元的首地址
#define uchar unsigned char
#define uint unsigned int
unsigned char TempH,TempL;
uint temp;
sbit KEY_INT=P3^2;
sbit RST=P1^0;
sbit P1_3=P1^3;
uchar num[9]={0x00,0x12};
unsigned char disp_buf[8]={0,0,0,0,0,0,0,0} ; //显示缓存
unsigned char rd[7]; //定义缓冲接收区
unsigned char nao[3]={0x00,0x00,0x00};
uchar year_moon,month_moon,day_moon,week; //农历声明
unsigned char KEY; //保存键值
int swich_date=0;
int i,j,k;
void Conversion(bit c,uchar year,uchar month,uchar day); //闹铃
unsigned char code TABLE[]={0x82,0x01,0x81,0x94,0x84,0xB4,0xA4,0x04,
0x82,0x01,0x81,0x94,0x84,0xC4,0xB4,0x04,
0x82,0x01,0x81,0xF4,0xD4,0xB4,0xA4,0x94,
0xE2,0x01,0xE1,0xD4,0xB4,0xC4,0xB4,0x04,
0x82,0x01,0x81,0x94,0x84,0xB4,0xA4,0x04,
0x82,0x01,0x81,0x94,0x84,0xC4,0xB4,0x04,
0x82,0x01,0x81,0xF4,0xD4,0xB4,0xA4,0x94,
0xE2,0x01,0xE1,0xD4,0xB4,0xC4,0xB4,0x04,0};
unsigned int code TABLE1[]={64260,64400,64524,64580,64684,64777,64820,64898,
64968,65030,65058,65110,65157,65178,65217 } ;
unsigned char th0_temp,tl0_temp;
//农历
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};
bit c_moon;
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);
}
}
}
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;
temp1=month/16;
temp2=month%16;
month=temp1*10+temp2;
temp1=day/16;
temp2=day%16;
day=temp1*10+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;
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-2099年) 调用函数示例:Conver_week(c_sun,year_sun,month_sun,day_sun)
如:计算2004年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(bit c,uchar year,uchar month,uchar day)
{ //c=0 为21世纪,c=1 为19世纪 输入输出数据均为BCD数据
uchar temp1,temp2;
temp1=year/16; //BCD->hex 先把数据转换为十六进制
temp2=year%16;
year=temp1*10+temp2;
temp1=month/16;
temp2=month%16;
month=temp1*10+temp2;
temp1=day/16;
temp2=day%16;
day=temp1*10+temp2;
if (c==0)
{
year+=0x64;
} //如果为21世纪,年份数加100
temp1=year/0x4; //所过闰年数只算1900年之后的
temp2=year+temp1;
temp2=temp2%0x7; //为节省资源,先进行一次取余,避免数大于0xff,避免使用整型数据
temp2=temp2+day+table_week[month-1];
if (year%0x4==0&&month<3) temp2-=1;
week=temp2%0x7;
}
//农历结束
unsigned char display_time(unsigned char *sd)
{
sd[0]=sd[0]&0x7f; //秒屏蔽保留位
sd[1]=sd[1]&0x7f; //分屏蔽保留位
sd[2]=sd[2]&0x3f; //时屏蔽保留位
disp_buf[0]=(sd[0]%16); disp_buf[1]=(sd[0]/16); // disp_buf[2]=31;
disp_buf[2]=(sd[1]%16); disp_buf[3]=(sd[1]/16); // disp_buf[5]=31;
disp_buf[4]=(sd[2]%16); disp_buf[5]=(sd[2]/16); disp_buf[6]=31;
disp_buf[7]=(sd[4]%16); ZLG7290_SendBuf(disp_buf,8); return 0;
}
unsigned char display_date(unsigned char *sd)
{
sd[0]=sd[0]&0x3f; //日屏蔽保留位
sd[2]=sd[2]&0x1f; //月屏蔽保留位
disp_buf[0]=(sd[0]%16); disp_buf[1]=(sd[0]/16); disp_buf[2]=(sd[2]%16)+0x80; //加小数点
disp_buf[3]=(sd[2]/16); disp_buf[4]=(sd[3]%16)+0x80; //加小数点
disp_buf[5]=(sd[3]/16); disp_buf[6]=0;
disp_buf[7]=2;
ZLG7290_SendBuf(disp_buf,8); return 0;
}
unsigned char display_daten(unsigned char *sd)
{
sd[0]=sd[0]&0x3f; //日屏蔽保留位
sd[2]=sd[2]&0x1f; //月屏蔽保留位
Conver_week(0,sd[3],sd[2],sd[0]); Conversion(0,sd[3],sd[2],sd[0]); sd[2]=month_moon;
sd[0]=day_moon;
sd[3]=year_moon;
disp_buf[0]=(sd[0]%16); disp_buf[1]=(sd[0]/16); disp_buf[2]=(sd[2]%16)+0x80; //加小数点
disp_buf[3]=(sd[2]/16); disp_buf[4]=(sd[3]%16)+0x80; //加小数点
disp_buf[5]=(sd[3]/16); disp_buf[6]=0;
disp_buf[7]=2;
ZLG7290_SendBuf(disp_buf,8); return 0;
}
void temperature_display() //处理,显示温度
{
uchar t;
if(temp&0x8000)
{
disp_buf[7]=0;//若温度为负,则显示负号
temp=~temp; // 取反加1
temp +=1;
}
TempH=temp>>4;
TempL=temp&0x0F;
TempL=TempL*6/10;//小数近似处理
t=TempH/100;
if(t==0) //如果百位为零,不显示
{
disp_buf[4] =31;
}
else
{
disp_buf[4] =TempH/100;
}
disp_buf[3]=(TempH%100)/10; //十位温度
disp_buf[2]=(TempH%100)%10; //个位温度,带小数点
disp_buf[1]=0x80;//显示小数点
disp_buf[0]=TempL;
ZLG7290_SendBuf(disp_buf,8);
}
unsigned char DelayNS(unsigned char no)
{
unsigned char i,j;
for(;no>0;no--)
for(i=0;i<100;i++)
for(j=0;j<100;j++); return 0;
}
//闹铃部分
void SING(unsigned char hi) {
th0_temp=(TABLE1[hi-1]/256); TH0=th0_temp;
tl0_temp=(TABLE1[hi-1]%256); TL0=tl0_temp;
TR0=1;
}
void DELAY1(unsigned char lo) {
unsigned char temp1,temp2; do
{
for(temp1=0;temp1<150;temp1++) for(temp2=0;temp2<200;temp2++);
}
while(lo--);
}
void INTT0() interrupt 1 {
TH0=th0_temp;
TL0=tl0_temp;
P1_3=~P1_3;
}
//闹铃部分
void clock()
{
if((rd[2]==nao[2])&&(rd[1]==nao[1])&&(rd[0]==nao[0]))
{
unsigned char i,hi,lo,coute,j=1;
TMOD= 0X01;
ET0 = 1;
EA = 1;
TR0 = 1;
// P1_3=1;
coute=0;
while(TABLE[coute]!=0) {
i=TABLE[coute];
coute++;
lo=i&0x0f;
hi=(i&0xf0)>>4;
if(hi>0)
SING(hi);
else
TR0=0;
DELAY1(lo);
}
}
k=((rd[2]&0xf0)>>4)*10+rd[2]&0x0f;
if ((rd[0]==0)&&(rd[1]==0)) { TR1=1;
for(j=0;j
3)
key1_count=0; }
if(KEY==14) //调整日期
{
key2_count++; if(key2_count>3) key2_count=1;
}
if(KEY==15) //调整时间
{
key3_count++; if(key3_count>4) key3_count=1;
}
if(KEY==5) //调整闹铃
{
key5_count++; if(key5_count>2) key5_count=1;
}
/*修改日期*/
if(key1_count==1) {
if(key2_count==1) //调整年
{
disp_buf[5] =31; disp_buf[4] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
rd[6]++;
if((rd[6]&0x0f)>=0x0a)
{
rd[6]&=0xf0;
rd[6]+=0x10;
}
}
// ISendStr(PCF8563,WRADDR,num,0x9);
}
else if(key2_count==2) //调整月 {
disp_buf[3] =31;
disp_buf[2] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{ rd[5]++;
if((rd[5]&0x0f)>=0x0a)
{
rd[5]&=0xf0;
rd[5]+=0x10;
}
if(rd[5]>0x12)
rd[5]=0x01;
}
// ISendStr(PCF8563,WRADDR,num,0x9);
}
else if(key2_count==3) //调整日
{ disp_buf[1] =31; disp_buf[0] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
rd[3]++;
if((rd[3]&0x0f)>=0x0a)
{
rd[3]&=0xf0;
rd[3]+=0x10;
}
if(rd[3]>=0x32) rd[3]=0x01;
}
// ISendStr(PCF8563,WRADDR,num,0x9);
}
}
/*修改时间*/
if(key1_count==0) {
if(key3_count==1) //修改小时
{
disp_buf[4] =31; disp_buf[5] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
rd[2]++;
if((rd[2]&0x0f)>=0x0a) {
rd[2]&=0xf0;
rd[2]+=0x10;
}
if(rd[2]>=0x24)
rd[2]=0x00;
}
//ISendStr(PCF8563,WRADDR,num,0x9);
}
else if(key3_count==2) //修改分钟
{ disp_buf[3] =31; disp_buf[2] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
rd[1]++;
if((rd[1]&0x0f)>=0x0a) {
rd[1]&=0xf0;
rd[1]+=0x10;
}
if((rd[1]&0xf0)>=0x60) rd[1]=0x00;
}
// ISendStr(PCF8563,WRADDR,num,0x9);
}
else if(key3_count==3) //修改秒
{ disp_buf[0] =31; disp_buf[1] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
rd[0]=0x00;
}
// ISendStr(PCF8563,WRADDR,num,0x9);
}
else if(key3_count==4) //修改星期
{ disp_buf[7] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
rd[4]++;
if(rd[4]>0x07)
rd[4]=0x01;
}
// ISendStr(PCF8563,WRADDR,num,0x9);
}
}
//修改闹铃
if(key1_count==3) {
if(key5_count==1) //修改小时
{
disp_buf[4] =31; disp_buf[5] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
nao[2]++;
if((nao[2]&0x0f)>=0x0a)
{
nao[2]&=0xf0;
nao[2]+=0x10;
}
if(nao[2]>=0x24) nao[2]=0x00;
}
//ISendStr(PCF8563,WRADDR,num,0x9);
}
else if(key5_count==2) //修改分钟
{ disp_buf[3] =31; disp_buf[2] =31;
ZLG7290_SendBuf(disp_buf,8);
DelayNS(4);
if(KEY==4)
{
nao[1]++;
if((nao[1]&0x0f)>=0x0a)
{
nao[1]&=0xf0;
nao[1]+=0x10;
}
if((nao[1]&0xf0)>=0x60) nao[1]=0x00;
}
}
}
for(i=0;i<7;i++)
{
num[i+2]=rd[i];
}
ISendStr(PCF8563,WRADDR,num,0x9);
DelayNS(1);
temperature_display();
}
if(key1_count==0)
{
display_time(rd);
}
else if(key1_count==1) display_date(rd+3); //调用显示日期子程序 else if(key1_count==3) // display_time(nao); temperature_display(); else
{
display_daten(rd+3); }
}
}