前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >单片机ds1302时钟程序(51单片机液晶显示程序)

单片机ds1302时钟程序(51单片机液晶显示程序)

作者头像
全栈程序员站长
发布2022-07-29 21:55:42
1.3K1
发布2022-07-29 21:55:42
举报
文章被收录于专栏:全栈程序员必看

大家好,又见面了,我是你们的朋友全栈君。

代码语言:javascript
复制
/*总体要求*/
 /*在1602上显示年月日星期时分秒,并且按照秒来实时更新显示
  可以闹钟设定,到点报警功能,报警响起时,任意键可以取消报警
  四个按键,根据功能可以调节参数,分别为功能键,数值增大键,数值减少键,闹钟查看键,
  每次按键按下,蜂鸣器都会滴一声,
  利用DS12C887实现断电后,再次上电,时间仍可以准确显示 */
  /*另外这个程序中文部分是学习了一个半月C语言的人的理解,难免有错误的地方
  如果你想移植程序,除了开头的各种脚需要修改,程序里面1602和DS12C887的写指令写数据,读指令读数据的IO口也要修改
  应该是有4~6处,一定要注意,不然你的程序不会亮起来,然后初始化部分,记得初始化后去掉,但是仿真的时候不要去掉,去掉没法显示*/
  /*非常非常非常重要的一个细节,总线上要加    AD[0..7]	 这个名字,如果没有永远也仿真不起来*/
#include<reg52.h>                         //52系列单片机机头文件
#define uchar unsigned char               //宏定义 定义一个无符号字符型数据
#define uint  unsigned int             	  //宏定义 定义一个无符号整型数据
sbit lcdrs=P2^0;                          //自定义数据/命令选择端口
sbit lcden=P2^1;                          //自定义使能信号
                                          /*P0口接液晶1602和时钟芯片DS12C887的AD0-AD7*/
sbit beep=P2^2;                           //蜂鸣器。有的按键接rd为低电平,就需要也定义一下rd,我这里面四个按键采用的是接地的结构!
sbit dsas=P2^3;							  //定义地址选通输入端 (读到那里去)
sbit dscs=P2^4;							  //定义芯片片选端	   (读哪个芯片)
sbit dsds=P2^5;							  //定义读数据允许输入脚,这里mot接地	(让不让读的开关)
sbit dsirq=P3^3;						  //定义中断请求输出
sbit dsrw=P2^6;							   //定义读允许输入	   (让读以后,我来输入数据)
sbit lcdrd=P2^7;						

sbit s1=P3^4;	                          //功能键	
sbit s2=P3^5;							  //数值增大键
sbit s3=P3^6;							  //数值减少键
sbit s4=P3^7;							  //闹钟查看键
										  //因为功能键要8次才能推出,这里可以加一个直接退出按键,可以放在外部中断上
bit    flag1, flag_ri;                                      //定义两个位变量
uchar  count,s1num,flag,t0_num;                             //定义几个无符号字符型数据,用到时再解释定义干啥用的
char   miao,shi,fen,year,month,day,week,amiao,afen,ashi;    //定义几个字符型数据,这个基本上都能理解。
uchar  code table[]=" 20  -  -      ";                          //让液晶固定显示的内容
uchar  code table1[]="      :  :  ";	                        //让液晶固定显示的内容
void   write_ds(uchar, uchar);			                    //函数申明	   寄存器初始化
void   set_alarm(uchar, uchar, uchar);	                    //函数申明	   设定闹钟
void   read_alarm();						                //函数申明	   读报警器
void   set_time();						                    //函数申明       寄存器初始化
void   delay(uint z)						                //延时函数
{										  
  uint x,y;								  //定义两个无符号整形数
  for(x=z;x>0;x--)						  //如果x>0,则y执行110次
     for(y=110;y>0;y--);				  //本小段为单位是1ms的延时
}
void di()								  //蜂鸣器报警
{
 beep=0;								  //低电平有效,蜂鸣器开始蜂鸣
 delay(100);							  //延时100毫秒,响0.1秒
 beep=1;								  //蜂鸣器停止鸣叫
}
void write_com(uchar com)				   //写液晶命令函数
{
  lcdrs=0;								   //rs低电平,表示要写命令
  lcden=0;								   //en低电平,表示为了下面送入数据,防止此时还是高电平,无法制造高脉冲
  P0=com;								   //D0-D7我接的是P0上面,数据送给它
  delay(3);								   //延时3ms
  lcden=1;								   //en高电平,延时三秒	再变成低电平,完成一次高脉冲,使数据送入液晶显示屏
  delay(3);								   //表示接下来,液晶显示屏可以过来读数据了
  lcden=0;
}
void write_date(uchar date)				   //写液晶数据函数
{
 lcdrs=1;								   //高电平表示要写数据
 lcden=0;								   //en低电平,表示为了下面送入数据,防止此时还是高电平,无法制造高脉冲
 P0=date;								   //D0-D7我接的是P0上面,数据送给它
 delay(3);								   //延时3ms
 lcden=1;								   //en高电平,延时三秒	再变成低电平,完成一次高脉冲,使数据送入液晶显示屏
 delay(3);								   //表示接下来,液晶显示屏可以过来读数据了
 lcden=0;
}
void init()                                //液晶初始化
{ 
  uchar  num;							   //定义一个字符
  EA=1;									   //打开全局中断控制
  EX1=1;								   //打开外部中断1中断
  IT1=1;								   //设置负跳变沿触发中断
  flag=0;								   // 自定义字符全部归零,为了方便实用 
  flag1=0;
  s1num=0;									//键盘按下的次数
  week=1;								   //星期显示最低也是周一的一
  lcden=0;								   //使能信号定义为零,表示还没有数据
  lcdrd=0;
  write_ds(0x0A,0x20);					   //首次使用ds12c887时使用,以后不必在写。打开振荡器
  write_ds(0x0B,0x26);					    // 设置24小时模式,数据二进制格式,开启闹钟中断。
                                            //0A,0B表示地址,20,26翻译成二进制是00100000,00100110对应着对着寄存器A和B从D7-D0 
  set_time();							   //设置默认上电时间
  write_com(0x38);						   //设置16x2显示,5x7点阵,8位数据接口
  write_com(0x0c);						   //设置开显示,不显示光标
  write_com(0x06);						   //写一个字符后,地址指针自动加1
  write_com(0x01);						   //显示清0,数据指针清0

  write_com(0x80);						   //设置数据指针位置,此处时数据指针第一行,第一处
  for(num=0;num<15;num++)				   //<15原因是第一行显示的数字或者空格一共14个
  {
    write_date(table[num]);				   //把要显示的数据送入到液晶的第一行
	delay(1);							   //延迟一毫秒
  }

    write_com(0x80+0x40);				   //设置数据指针位置,此处时数据指针第二行,第一处
    for(num=0;num<11;num++)				   //<11原因是第二行显示的数字或者空格一共10个
  {
    write_date(table1[num]);				   //把要显示的数据送入到液晶的第二行
	delay(1);							   //延迟一毫秒
  }
}
void write_sfm(uchar add,char date)		   //1602液晶刷新时分秒函数4为时,7为分,10为秒	  ,4.7.10均是显示位置
{   
   char shi,ge;							   //定义两个数,来表示十位数和个位数
   shi=date/10;							   // 要送去显示的十位
   ge=date%10;							   //要送去显示的个位  
   write_com(0x80+0x40+add);				   //时间是在第二行显示,所以是加0X40,在add处显示
   write_date(0x30+shi);				   //shi和ge都是整形数值,但是液晶1602是内置ASCII表格,0在第048位
   write_date(0x30+ge);					   //转成十六进制就是0x30,
}
/*这个地方可能会有人迷糊,设置时分秒,怎么只设置十位和个,好像给人的感觉就是只设置时位,和分位,秒位怎么不设置,
实际上是,这里写入并显示的是  把时的十位和个位,分的十位和个位,秒的十位和个位,分别送去进行运算 ,然后先写命令确定位置,
再写数据,用来显示,下面一个子函数也是同样的道理*/
void write_nyr(uchar add,char date)		   //1602液晶刷新年月日函数,3为年,6为月9为天	 ,369也分别是位置
{   
   char shi,ge;							   //定义两个数,来表示十位数和个位数
   shi=date/10;							   // 要送去显示的十位
   ge=date%10;							   //要送去显示的个位  
   write_com(0x80+add);			           //时间是在第二行显示,所以是加0X40,在add处显示
   write_date(0x30+shi);				   //shi和ge都是整形数值,但是液晶1602是内置ASCII表格,0在第048位
   write_date(0x30+ge);					   //转成十六进制就是0x30,
}   /*实际上到了主函数,时分秒,年月日会有六个子函数进行分别调用,这里用两个子函数来表示,不能用一个的原因是显示的内容不在一行,
主要是写入命令的位置不一样,大家可以观察一下*/
void write_week(char we)					//写液晶1602星期显示函数   
{
  write_com(0x80+12);						//让we在0x80+12这个位置显示
  switch(we)								//多分支结构的条件选择语句,we等于几就去执行几号分支
  {
   case 1:write_date('M');				   //写入M。整个的意思是如果we等于1,则显示MON,即星期一
          delay(5);						   //延时5毫秒后写入O
		  write_date('O');				   //写入O
          delay(5);						   //延时5毫秒后写入N
  	      write_date('N');				   //写入N
          break;						   //间断,为了和后面分开
   case 2:write_date('T');				   //写入T。整个的意思是如果we等于2,则显示TUE,即星期二
          delay(5);						   //延时5毫秒后写入U
		  write_date('U');				   //写入U
          delay(5);						   //延时5毫秒后写入E
  	      write_date('E');				   //写入E
          break;						   // 间断,为了和后面分开
   case 3:write_date('W');				   //写入W。整个的意思是如果we等于3,则显示WED,即星期三
          delay(5);						   //延时5毫秒后写入E
		  write_date('E');				   //写入E
          delay(5);						   //延时5毫秒后写入D
  	      write_date('D');				   //写入D
          break;						   //间断,为了和后面分开
   case 4:write_date('T');				   //写入T。整个的意思是如果we等于4,则显示THU,即星期四
          delay(5);						   //延时5毫秒后写入H
		  write_date('H');				   //写入H
          delay(5);						   //延时5毫秒后写入U
  	      write_date('U');				   //写入U
          break;						   //间断,为了和后面分开
   case 5:write_date('F');				   //写入F。整个的意思是如果we等于5,则显示FRI,即星期五
          delay(5);						   //延时5毫秒后写入R
		  write_date('R');				   //写入R
          delay(5);						   //延时5毫秒后写入I
  	      write_date('I');				   //写入I
          break;						   //间断,为了和后面分开
   case 6:write_date('S');				   //写入S。整个的意思是如果we等于6,则显示SAT,即星期六
          delay(5);						   //延时5毫秒后写入A
		  write_date('A');				   //写入A
          delay(5);						   //延时5毫秒后写入T
  	      write_date('T');				   //写入T
          break;						   //间断,为了和后面分开
   case 7:write_date('S');				   //写入S。整个的意思是如果we等于7,则显示SUN,即星期日
          delay(5);						   //延时5毫秒后写入U
		  write_date('U');				   //写入U
          delay(5);						   //延时5毫秒后写入N
  	      write_date('N');				   //写入N
          break;						   //间断,为了和后面分开
	
   }
}
void keyscan()							   //数字时钟按键扫描函数
{
   if(flag_ri==1)						   //如果标志位,闹钟等于1,则蜂鸣器会响 ,这一句和按键扫描没有关系,主要是为了任意键可以消除
                                           //闹钟的报警,可以理解为,为闹钟取消而增加的一个扫描小程序。
   {
	if((s1==0)||(s2==0)||(s3==0)||(s4==0))	//如果s1-s4有一个为零,低电平	 ||这个符号是关系运算符, 或的意思,有一个是低电平,则进行下一步。
	{
	 delay(5);								//延时去抖动
	 if((s1==0)||(s2==0)||(s3==0)||(s4==0))	//再次确认,如果s1-s4有一个为零,低电平
	 {
	   while(!(s1&&s2&&s3&&s4));			/*/如果他们运算的结果为0	 ,当闹钟响起,就会进入这个按键扫描程序,这没有按下键盘的时候,
	   就会运行到这里,进行判断循环,一直等在,四个按键,有一个为0,这个语句就会判断为假,进行下一步运算,&&这个符号是关系运算的 与,
	   一个值为零,则整体结果运算为0*/
	   di();						    	//则蜂鸣器会响,闹钟有自己的蜂鸣程序,这里有个DI主要原因是,每次按键 都会响,这是事先规定号的
	   flag_ri=0;							//清除报警标志
	 }
	}
   }

  if(s1==0)									  //检测按键s1
  {
   delay(5);								  //延时5毫秒,也可叫延时去抖动。
   if(s1==0)								  //再次检测按键s1
   {
    s1num++;								  //定义为键盘按下的次数
	if(flag1 == 1)							  //这个是用来闹钟的,如果等予1,说明现在是在设置闹钟值,从下面来看这个只能调节每天的闹钟,不能调节年月
	   if(s1num == 4)						 //因为只能调节时分秒,让他们在里面循环
	      s1num=1;							  //这个地方我卡了几天,大家要综合前后文,明白flag1是闹钟被按下了,flag是为了跳出主程序循环,为了设置时间
	flag=1;	    							  //按键计数以后,则这个位变量变成1,
	while(!s1);								  /*!s1 的意思是取s1相反的数,是用来测试判断按键按下后有没有松开,松开则计数一次后,电平
	                     					  变成高电平,说明此时已经松开,也确定了s1num的具体数值,也就是按了几下*/
	di();									  //每按下一次则响一声
	switch(s1num)							  //判断是按了几下,则选择什么位置光标点闪烁。
	{
		case 1:write_com(0x80+0x40+10);		  //很明显这是显示屏第二行的显示指令,10  说明光标调节的是秒
		       write_com(0x0f);     		  /*/不管你按的再快,光标也会从第一步开始,所以把 开显示,显示光标,光标闪烁  这个lcd的这个
											     指令设置放在这里	*/
			   break;						  //会不停的跳出,再去问现在是按了几下了,按了几下,就会进入对应的里面,下面都是一样的不解释了。
		case 2:write_com(0x80+0x40+7);     	  /*这个就是光标在分钟上闪烁,(不光时分秒都是在他们的十位闪烁,
		                                       如果送的是两位数,LCD的指针可以自动加一,把个位也给显示了)*/
			   break;
		case 3:write_com(0x80+0x40+4);    	   //这个是在小时的位置显示闪烁
			   break;
		case 4:write_com(0x80+12);   			//这就是进入显示屏第一行了,在显示星期的位置,第十位,最右边,是调节星期的位置。
			   break;
		case 5:write_com(0x80+9);    			 //日
			   break;
		case 6:write_com(0x80+6);    			 //月
			   break;
		case 7:write_com(0x80+3);     			  //年
			   break;
		case 8:s1num=0;							  //把按键数置零,等这个小程序运行完,则光标不在闪缩,一切正常运行。相当于不选择任何位置。
		       write_com(0x0c);					  //设置开显示,不显示光标
			   flag=0;							  //等于0就是回到主程序的正常运行(可以参看以下主程序,更能方便理解)
				/*下面的几句基本一样,明白一个就全明白了,这个地方主要是查表,查寄存器A,功能列表有时分秒,闹钟的也有时分秒,年月日他们分别
				对应的寄存器地址,比如秒对应的是00H,分钟对应的是02H,所以0,2,4,6,7,8,9都是地址,往这个里面写,就是写的时分秒,
				可能有人问,既然是00H,02H,为什么写成0,2等等,其实是因为十六进制和十进制在0~9,书写虽然有区别,但是结果都一样*/
			   write_ds(0,miao);				   
			   write_ds(2,fen);
			   write_ds(4,shi);
			   write_ds(6,week);
			   write_ds(7,day);
			   write_ds(8,month);
			   write_ds(9,year);
			   break;	                           //跳出
	}
   }
  }
  if(s1num!=0)	                                    //测试不等于 0,也就是只有当s1按下时才能检测和用来启动s2和s3.
  {
   if(s2==0)                               	        //问S2按键有没有按下
   {
	delay(1);	 									//延时1毫秒
	if(s2==0)	 									//再次来检测
	{
	  while(!s2); 									 //给s2取反
	  di();											 //滴一声
	  switch(s1num)									 //根据s1按下的次数,来确定调节哪个位置
	  {				   								 //下面给大家解释一组,应该可以全部明白
	   case 1:miao++;								 //按键s2按一下,这里就会加一	  设置秒钟
	          if(miao==60) 							 //如果加到了60 ,秒就会变成0
			     miao=0;
		     write_sfm(10,miao);	                 //把秒的值经过运算,送到LCD相应的位置去显示
			 write_com(0x80+0x40+10);                /*这一句是调用写指令,因为指针把秒显示完毕后,光标就会在秒的个位,
			                                         所以这一句是把光标在拉回来,因为不能正好一下子就把秒的值设置的正好*/
			 break;					                 //然后跳出,等待按键再次按下
	   case 2:fen++;					             //设置分钟
	          if(fen==60)				             //分,秒都用到60,生活中并没有60,这里也不能设置成59,但是可以改程序为>59.道理是一样的
			     fen=0;
			 write_sfm(7,fen);
			 write_com(0x80+0x40+7);
			 break;
	   case 3:shi++;					              //设置时钟
	          if(shi==24)				               //和上面一样的道理
			     shi=0;
			  write_sfm(4,shi);
			  write_com(0x80+0x40+4);
			  break;
	   case	4:week++;				                  //设置星期
	          if(week ==8)				              //当按下第八次的时候,则8就会变成1
			     week=1;
			  write_week(week);
			  write_com(0x80+12);
			  break;
	  case 5:day++;					                    //设置天数
	         if(day==32)			  
			    day=1;
			 write_nyr(9,day);
			 write_com(0x80+9);
			 break;
	 case 6:month++;			                       //设置月数
	        if(month==13)
			   month=1;
			write_nyr(6,month);
			write_com(0x80+6);
			break;
	case 7:year++;					                    //设置年
	       if(year==100)
		      year=0;
			 write_nyr(3,year);
			 write_com(0x80+3);
			 break;	  
	  }
    }
   }
   if(s3==0)	                                       //上面是增大键,下面就是减少键
   {
	delay(1);	                                       //延迟1毫秒
	if(s3==0)	                                      //再次判断有没有按下
	{
	  while(!s3);                                     //取相反数,看看是不是为真
	  di();
	  switch(s1num)	                                  //根据按键1,就可以确定光标在什么位置,下面介绍和上面基本上一样,
	  {					                              //只不过没有60了,反而有59,可以好好理解一下,其实很简单这个地方,就不解释了
	   case 1:miao--;				                   //设置秒钟
	          if(miao==-1)
			     miao=59;
		     write_sfm(10,miao);
			 write_com(0x80+0x40+10);
			 break;
	   case 2:fen--;				                  //设置分钟
	          if(fen==-1)
			     fen=59;
			 write_sfm(7,fen);
			 write_com(0x80+0x40+7);
			 break;
	   case 3:shi--;					              //设置小时
	          if(shi==-1)
			     shi=23;
			  write_sfm(4,shi);
			  write_com(0x80+0x40+4);
			  break;
	   case	4:week--;					              //设置星期
	          if(week==-0)
			     week=7;
			  write_week(week);
			  write_com(0x80+12);
			  break;
	   case 5:day--;					              //设置天数
	          if(day==0)
			     day=31;
		 	  write_nyr(9,day);
			  write_com(0x80+9);
			  break;
	   case 6:month--;					               //设置月数
	         if(month==0)
			    month=12;
			 write_nyr(6,month);
			 write_com(0x80+6);
			 break;
 	   case 7:year--;					                 //设置年
	         if(year==-1)
		      year=99;
			 write_nyr(3,year);
			 write_com(0x80+3);
			 break;	  
	  }
    }
   }
  }
  if(s4==0)	                                                 //这个地方是闹钟设置键
  {
   delay(5);	                                             //延时5毫秒
   if(s4==0)	                                             //再次判断是否按下
   {
	flag1=~flag1;                                            //如果按下,则按键1程序,进入时分秒的循环,这里的讲解和前面相呼应。
	while(!s4);	                                             //取反是不是为真
	di();		                                             //为真说明,有按键按下,响一声,
	s1num=0;												 
	/*调试仿真时发现如果按下了按键一  在按闹钟,然后取消闹钟,在按按键一,会不出现光标,为了让闹钟和按键一不冲突,我添加这一句
	每当我按下按键一,不管按几次,只要按下按键4,都会让光标变成0位 */
	if(flag1==0)                                             //如果等于0,说明按得第二下,属于退出闹钟设置。
	{
	 flag=0;												 //程序运行到这一步,说明运行完这个程序,就要回到主程序了,
	                                                         //把按键1标志指令以后,本程序结束,可以进入主程序,否则还要在去按键1,按几下     
	 write_com(0x80+0x40);									 //把指针选择第二行,第一位,
	 write_date(' ');										 //写入两个空的符号,主要是为了区别状态,是调整时间还是设置闹钟
	 write_date(' ');
	 write_com(0x0c);										  //关闭光标闪烁
	 write_ds(1,miao);										  //把时分秒送到DS12C887,程序正常运行
	 write_ds(3,fen);										  //这个时候的135要注意,可以看出是新设置闹钟值
	 write_ds(5,shi);
	}
	else													   //如果flag不等于0,则进入这个程序,实际上程序都是先运行这个
	{														   //进入闹钟设置
	 read_alarm();											   //读闹钟原来的数据
	 miao=amiao;											   //分别送入时分秒
	 fen=afen;
	 shi=ashi;
	 write_com(0x80+0x40);									   //写指令,把指针,也就是光标位置放在第二排第一位
	 write_date('R');										   //分别送入R  I送去显示,可以区分现在是在调整时间,还是修改闹钟
	 write_date('I');
	 write_com(0x80+0x40+3);								   /*这个地方还有一个指针,就会让很多人不理解,其实这个地方设计很巧妙
	                                                           3号位置,正好是时钟前面一位,然后再那边闪烁,接着等待按键1,来选择调整时分秒,
															   按一下变换一个位置,在3号位闪烁还不影响开始时的闹钟值*/
	                                                            
	 write_sfm(4,ashi);										   //送到液晶去显示
	 write_sfm(7,afen);
	 write_sfm(10,amiao);
	}
   }
  } 
}
void write_ds(uchar add,uchar date)	                           //写DS12C887函数 
{
 dscs=0;													   //根据时序图,把相应的脚变化高低,片选端,低电平有效
 dsas=1;													   //地址选通输入端
 dsds=1;													   //读允许输入脚
 dsrw=1;														//写输入脚
 P0=add;														//写地址
 dsas=0;
 dsrw=0;
 P0=date;													    //写命令
 dsrw=1;
 dsas=1;
 dscs=1;
}
uchar read_ds(uchar add)									   //读DS12C887函数	,这里没有用void要注意
{
 uchar ds_date;												   //定义一个无符号函数
 dsas=1;													   //根据时序图,进行高低电平设置
 dsds=1;
 dsrw=1;
 dscs=0;
 P0=add;													   //写地址,确定要读什么位置的值
 dsas=0;
 dsds=0;
 P0=0xff;													   //先把P1进行置高,清理数据
 ds_date=P0;													//把新的得到了  12c887里面的数值,赋值给ds_date
 dsds=1;														//根据时序图继续设置电平
 dsas=1;
 dscs=1;
 return ds_date;											   //谁调用它,什么位置调用的,就会把这个新得到的值在返回给他,
															   //看得出来只有读闹钟值是调用了。
}
 void set_time()											   //首次操作12c887时给寄存器初始化,完成后,要把这一段用 给屏蔽起来
{
 write_ds(0,0);												  //根据寄存器A,相当于	所有的位置,全部显示00。
 write_ds(1,0);
 write_ds(2,0);
 write_ds(3,0);
 write_ds(4,0);
 write_ds(5,0);
 write_ds(6,0);
 write_ds(7,0);
 write_ds(8,0);
 write_ds(9,0);
} 	
void read_alarm()											  //读取闹钟值
{
 amiao=read_ds(1);											   //根据寄存器A,分别把闹钟值读出来,并且赋值给时分秒
 afen=read_ds(3);
 ashi=read_ds(5);
}
void main()													 //主程序
{
 init();													  //调用初始化函数
 while(1)													  //为真不断循环,等待闹钟中断
 {
 keyscan();													   //按键子程序,不断地扫描
 if(flag_ri==1)												   //如果有闹钟,则进入这里
 {
  di();
  delay(100);													//一直响 滴滴 
  di();
  delay(500);
 }
 if((flag==0) && (flag1==0))								   //如果按键标志位和闹钟标志位都为0,则继续
 {
  keyscan();												   //按键继续扫描
  year=read_ds(9);											   //把DS12C887里面的数值不断读取
  month=read_ds(8);
  day=read_ds(7);
  week=read_ds(6);
  shi=read_ds(4);
  fen=read_ds(2);
  miao=read_ds(0);
  write_sfm(10,miao);										   //这里把DS12c887的值不断地写入到lcd1602
  write_sfm(7,fen);
  write_sfm(4,shi);
  write_week(week);
  write_nyr(3,year);
  write_nyr(6,month);
  write_nyr(9,day);
 }
 }
}															 
void exter()interrupt 0									    //中断1程序
{
 uchar c;														//定义一个无符号数值
 flag_ri=1;														//闹钟中断时间到了
 c=read_ds(0x0c);												//读取c寄存器的值,表示响应了中断
}

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/129766.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年4月1,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
云开发 CloudBase
云开发(Tencent CloudBase,TCB)是腾讯云提供的云原生一体化开发环境和工具平台,为200万+企业和开发者提供高可用、自动弹性扩缩的后端云服务,可用于云端一体化开发多种端应用(小程序、公众号、Web 应用等),避免了应用开发过程中繁琐的服务器搭建及运维,开发者可以专注于业务逻辑的实现,开发门槛更低,效率更高。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档