驱动DS1302实时时钟模块,并通过数码管显示时间、日期和秒数

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.文件

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值