1.补充知识点
代码中用到的地方会有下标
知识点①:
在二进制系统中,左移操作可以通过乘以2的幂来实现。对于将一个数左移4位,可以通过乘以 (2^4 = 16) 来实现。这是因为每次左移1位,相当于乘以2;左移4位就是乘以 (2^4)(即16)
假设 tmp 的值是4(二进制表示为 0000 0100)。我们要将这个值左移4位:
左移4位:
初始值:0000 0100
左移4位后:0100 0000
乘以16:
4 * 16 = 64
二进制表示为 0100 0000(这是64的二进制表示)
这两种方法的结果是一样的。乘以16就是将二进制数左移4位,得到同样的结果。
知识点②:
具体来说,DS1302的秒寄存器(ds1302_sec_add)的高位(bit 7)用来控制时钟的启停状态:
bit 7(CH):当 CH 位被设置为1时,DS1302的时钟暂停。这意味着时钟停止更新秒、分、时等时间数据,并且减少功耗,以延长电池寿命或减少功耗。这种状态通常用于在电池供电时节省能量,或者在需要时暂停时钟以执行特定的时间设置或校准操作。
因此,通过向 ds1302_sec_add 寄存器写入 0x80,即设置 CH 位为1,可以暂停DS1302的时钟运行。
知识点③:
BCD处理就是将十进制转换为特定的二进制 --------->为什么用BCD?DS1302 使用BCD 格式来存储时间数据。
在普通二进制表示中,十进制数 59 用二进制表示为 111011。
在 BCD 表示中,十进制数 59 被拆分为两个 BCD 数位:
十位 5 的 BCD 是 0101
个位 9 的 BCD 是 1001
所以,BCD 表示的 59 是 0101 1001。
2.主要代码
main.c
#define KeyPort P3 //定义按键端口
#define DataPort P0 //定义数据端口 程序中遇到DataPort 则用P0 替换
sbit LATCH1=P2^2;//定义锁存使能端口 段锁存
sbit LATCH2=P2^3;// 位锁存
bit ReadTimeFlag;//定义读时间标志
unsigned char code dofly_DuanMa[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};// 显示段码值0~9
unsigned char code dofly_WeiMa[]={0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f};//分别对应相应的数码管点亮,即位码
unsigned char TempData[8]; //存储显示值的全局变量
void DelayUs2x(unsigned char t);//us级延时函数声明
void DelayMs(unsigned char t); //ms级延时
void Display(unsigned char FirstBit,unsigned char Num);//数码管显示函数
unsigned char KeyScan(void);//键盘扫描
void Init_Timer0(void);//定时器初始化
/*------------------------------------------------
主函数
------------------------------------------------*/
void main (void)
{
unsigned char num,displaynum;
Init_Timer0();
Ds1302_Init();
Ds1302_Write_Time(); //用于向DS1302实时时钟模块写入当前系统时间
while (1) //主循环
{
num=KeyScan(); //按键扫描
if(num==1)
{
displaynum++;
if(displaynum==3)
displaynum=0;
}
if(ReadTimeFlag==1)
{
ReadTimeFlag=0;
Ds1302_Read_Time();
if(displaynum==0) //显示时间
{
TempData[0]=dofly_DuanMa[time_buf1[4]/10];//时 //数据的转换,
TempData[1]=dofly_DuanMa[time_buf1[4]%10];//因我们采用数码管0~9的显示,将数据分开
TempData[2]=0x40; //加入"-"
TempData[3]=dofly_DuanMa[time_buf1[5]/10];//分
TempData[4]=dofly_DuanMa[time_buf1[5]%10];
TempData[5]=0x40;
TempData[6]=dofly_DuanMa[time_buf1[6]/10];//秒
TempData[7]=dofly_DuanMa[time_buf1[6]%10];
}
else if(displaynum==1)//显示日期
{
TempData[0]=dofly_DuanMa[time_buf1[1]/10];//年
TempData[1]=dofly_DuanMa[time_buf1[1]%10];
TempData[2]=0x40; //加入"-"
TempData[3]=dofly_DuanMa[time_buf1[2]/10];//月
TempData[4]=dofly_DuanMa[time_buf1[2]%10];
TempData[5]=0x40;
TempData[6]=dofly_DuanMa[time_buf1[3]/10];//日
TempData[7]=dofly_DuanMa[time_buf1[3]%10];
}
else if(displaynum==2)//显示周 秒
{
TempData[0]=0x40;
TempData[1]=dofly_DuanMa[time_buf1[7]%10];//周
TempData[2]=0x40; //加入"-"
TempData[3]=0;
TempData[4]=0;
TempData[5]=0;
TempData[6]=dofly_DuanMa[time_buf1[6]/10];//秒
TempData[7]=dofly_DuanMa[time_buf1[6]%10];
}
}
}
}
/*------------------------------------------------
uS延时函数,含有输入参数 unsigned char t,无返回值
unsigned char 是定义无符号字符变量,其值的范围是
0~255 这里使用晶振12M,精确延时请使用汇编,大致延时
长度如下 T=tx2+5 uS
------------------------------------------------*/
void DelayUs2x(unsigned char t)
{
while(--t);
}
/*------------------------------------------------
mS延时函数,含有输入参数 unsigned char t,无返回值
unsigned char 是定义无符号字符变量,其值的范围是
0~255 这里使用晶振12M,精确延时请使用汇编
------------------------------------------------*/
void DelayMs(unsigned char t)
{
while(t--)
{
//大致延时1mS
DelayUs2x(245);
DelayUs2x(245);
}
}
/*------------------------------------------------
显示函数,用于动态扫描数码管
输入参数 FirstBit 表示需要显示的第一位,如赋值2表示从第三个数码管开始显示
如输入0表示从第一个显示。
Num表示需要显示的位数,如需要显示99两位数值则该值输入2
------------------------------------------------*/
void Display(unsigned char FirstBit,unsigned char Num)
{
static unsigned char i=0;
DataPort=0; //清空数据,防止有交替重影
LATCH1=1; //段锁存
LATCH1=0;
DataPort=dofly_WeiMa[i+FirstBit]; //取位码
LATCH2=1; //位锁存
LATCH2=0;
DataPort=TempData[i]; //取显示数据,段码
LATCH1=1; //段锁存
LATCH1=0;
i++;
if(i==Num)
i=0;
}
/*------------------------------------------------
定时器初始化子程序
------------------------------------------------*/
void Init_Timer0(void)
{
TMOD |= 0x01; //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响
//TH0=0x00; //给定初值
//TL0=0x00;
EA=1; //总中断打开
ET0=1; //定时器中断打开
TR0=1; //定时器开关打开
}
/*------------------------------------------------
定时器中断子程序
------------------------------------------------*/
void Timer0_isr(void) interrupt 1
{
static unsigned int num;
TH0=(65536-2000)/256; //重新赋值 2ms
TL0=(65536-2000)%256;
Display(0,8); // 调用数码管扫描
num++;
if(num==50) //大致100ms
{
num=0;
ReadTimeFlag=1; //读标志位置1
}
}
/*------------------------------------------------
按键扫描函数,返回扫描键值
------------------------------------------------*/
unsigned char KeyScan(void)
{
unsigned char keyvalue;
if(KeyPort!=0xff) //是否不等于 0xff 来判断是否有按键按下
{
DelayMs(10); //有按键按下,则进行延时 DelayMs(10),等待按键稳定 ----->消抖
if(KeyPort!=0xff) //再次检查 KeyPort 的值是否不等于 0xff,确保按键状态稳定
{
keyvalue=KeyPort;
while(KeyPort!=0xff); //在按键释放前持续等待
switch(keyvalue) //如果检测到按键按下且按键值在预期范围内,返回对应的按键值(1 到 8)
{
case 0xfe:return 1;break;
case 0xfd:return 2;break;
case 0xfb:return 3;break;
case 0xf7:return 4;break;
case 0xef:return 5;break;
case 0xdf:return 6;break;
case 0xbf:return 7;break;
case 0x7f:return 8;break;
default:return 0;break; //如果 keyvalue 不匹配上述任何值,则返回 0,表示没有按下有效的按键
}
}
}
return 0;
}
DS1302.c
unsigned char time_buf1[8] = {20,10,6,5,12,55,00,6};//空年月日时分秒周
unsigned char time_buf[8] ; //空年月日时分秒周
/*------------------------------------------------
向DS1302指定地址addr写入一字节数据d
------------------------------------------------*/
void Ds1302_Write_Byte(unsigned char addr, unsigned char d)
{
unsigned char i;
RST_SET; // 设置RST引脚为高电平 这是复位引脚,高电平时芯片复位到初始状态,复位后DS1303的内部状态喝寄存器将清除
//按位发送一个地址 addr,通过设置数据线的电平和通过 SCK(时钟线)来控制数据的传输时序,以实现地址的发送。
//写入目标地址:addr
addr = addr & 0xFE; //最低位置零 强制将DS1302地址的最低位设为0是为了确保写入操作能够顺利进行,符合DS1302的通信协议规范。
for (i = 0; i < 8; i ++) //循环次数为8次(因为地址通常是8位),用于逐位发送地址信息
{
if (addr & 0x01) //如果addr的最后一位是1
{
IO_SET; //则电平置1
}
else //否则
{
IO_CLR;//电平置0
}
SCK_SET; //将时钟线(SCK)置为高电平,表示数据线上的数据可以被读取或者传输
SCK_CLR; //将时钟线(SCK)置为低电平,数据传输的完成
addr = addr >> 1; //将 addr 向右移动一位,即将下一个要发送的位移动到最低位,为下一轮循环做准备
}
//写入数据:d
for (i = 0; i < 8; i ++)
{
if (d & 0x01)
{
IO_SET;
}
else
{
IO_CLR;
}
SCK_SET;
SCK_CLR;
d = d >> 1;
}
RST_CLR; //停止DS1302总线 这是一个复位清除引脚,当将RST_CLR引脚置高时,会清除DS1303上的复位状态,使其恢复正常工作状态,不再处于复位状态。
}
/*------------------------------------------------
从DS1302指定地址addr读出一字节数据temp
------------------------------------------------*/
unsigned char Ds1302_Read_Byte(unsigned char addr)
{
unsigned char i;
unsigned char temp;
RST_SET;
//写入目标地址:addr
addr = addr | 0x01;//最低位置高
for (i = 0; i < 8; i ++)
{
if (addr & 0x01)
{
IO_SET;
}
else
{
IO_CLR;
}
SCK_SET;
SCK_CLR;
addr = addr >> 1;
}
//输出数据:temp
for (i = 0; i < 8; i ++)
{
temp = temp >> 1;
if (IO_R)
{
temp |= 0x80;
}
else
{
temp &= 0x7F;
}
SCK_SET;
SCK_CLR;
}
RST_CLR; //停止DS1302总线
return temp;
}
/*------------------------------------------------
向DS1302写入时钟数据
------------------------------------------------*/
void Ds1302_Write_Time(void)
{
unsigned char i,tmp;
for(i=0;i<8;i++)
{ //BCD处理
tmp=time_buf1[i]/10; // 取得十位数
time_buf[i]=time_buf1[i]%10; // 取得个位数
time_buf[i]=time_buf[i]+tmp*16;// 将十位数左移4位并与个位相加,得到BCD码 tmp * 16 在二进制表示中等价于将 tmp 左移4位。 (知识点①)
}
Ds1302_Write_Byte(ds1302_control_add,0x00); //关闭写保护
/*
DS1302 实时时钟模块中,向 ds1302_control_add 寄存器地址写入 0x00 的操作通常用于关闭写保护功能,即允许对控制寄存器和时钟数据寄存器进行写操作,而不会受到保护。
*/
Ds1302_Write_Byte(ds1302_sec_add,0x80); //暂停 在DS1302实时时钟模块中,写入 0x80 到 ds1302_sec_add 寄存器地址可以暂停时钟的运行。这个操作实际上是在控制DS1302时钟的启停。(知识点②)
//Ds1302_Write_Byte(ds1302_charger_add,0xa9); //涓流充电
Ds1302_Write_Byte(ds1302_year_add,time_buf[1]); //年
Ds1302_Write_Byte(ds1302_month_add,time_buf[2]); //月
Ds1302_Write_Byte(ds1302_date_add,time_buf[3]); //日
Ds1302_Write_Byte(ds1302_day_add,time_buf[7]); //周
Ds1302_Write_Byte(ds1302_hr_add,time_buf[4]); //时
Ds1302_Write_Byte(ds1302_min_add,time_buf[5]); //分
Ds1302_Write_Byte(ds1302_sec_add,time_buf[6]); //秒
Ds1302_Write_Byte(ds1302_day_add,time_buf[7]); //周
Ds1302_Write_Byte(ds1302_control_add,0x80); //打开写保护
}
/*------------------------------------------------
从DS1302读出时钟数据
------------------------------------------------*/
void Ds1302_Read_Time(void)
{
unsigned char i,tmp;
time_buf[1]=Ds1302_Read_Byte(ds1302_year_add); //年
time_buf[2]=Ds1302_Read_Byte(ds1302_month_add); //月
time_buf[3]=Ds1302_Read_Byte(ds1302_date_add); //日
time_buf[4]=Ds1302_Read_Byte(ds1302_hr_add); //时
time_buf[5]=Ds1302_Read_Byte(ds1302_min_add); //分
time_buf[6]=(Ds1302_Read_Byte(ds1302_sec_add))&0x7F;//秒
time_buf[7]=Ds1302_Read_Byte(ds1302_day_add); //周
for(i=0;i<8;i++)
{ //BCD处理 (知识点③)
tmp=time_buf[i]/16;
time_buf1[i]=time_buf[i]%16;
time_buf1[i]=time_buf1[i]+tmp*10;
}
/*------------------------------------------------
DS1302初始化
------------------------------------------------*/
void Ds1302_Init(void)
{
RST_CLR; //RST脚置低
SCK_CLR; //SCK脚置低
Ds1302_Write_Byte(ds1302_sec_add,0x00);
}
3.文件