找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 15050|回复: 30
打印 上一主题 下一主题
收起左侧

红外遥控单片机数码管时钟制作

  [复制链接]
跳转到指定楼层
楼主
ID:76686 发表于 2015-4-11 22:35 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
      初学单片机的菜鸟们,都很想自己制作一款单片机电子钟,放在自己的电脑桌上,展现一下学习单片机的成果,现在就来讲一讲制作这款遥控单片机数码管时钟的过程。单片机采用STC89C52RC单片机,时钟芯片采用DS1302控时,18B20采集温度,红外遥控(通用电视机顶盒遥控器),万年历显示,一路闹钟,自动/手动切换显示,超低功耗掉电工作模式(适合电池供电),拥有备用电源掉电不掉时。数码管经消影处理,亮度显示均匀,两个二位一体数码管是互相倒过来安装的,从而让两个小数点在中间形成一个冒号。完整程序下载: 遥控单片机数码管时钟程序(无闹钟版).rar (47.93 KB, 下载次数: 292)




       测量温度时的情况:




        制作一款单片机电子钟需要那些元器件呢?请看下图,所需要的元件名称、外形及数量都在图中。



        数码管的选择,本作品要选用的是二位一体共阳极0.56寸数码管,下图有两种二位一体共阳极数码管,但左边的共有18个引脚,右边的只有10个,为了方便布线,我们选择右边的10引脚哪种数码管。数码管的引脚,本作品将两个数码管相倒安装,所以其引脚顺序也不一样了,请对照下面两幅图片,安装时看清认准。







        这里就不给出电原理图,而直接给出实物连接图,下面给出的图,布的是双层板,新手们先从单层的练起,然后再缩小布局,采用双层板。图中蓝色线表示洞洞板的布线,红色线表示顶层线。安装时整板是根据洞洞板来布局的,焊接时,红线作为飞线来布,图中的Z1表示接线端子,H1是一体化红外接受头,数码管一定是共阳极数码管,焊接时先焊飞线,再焊元件。数码管的四角都有突起,所以放在洞洞板上与板面之间会有缝隙,可以将飞线隐藏在数码管的下面。本作品中的DS1302没有装备用电池。本程序加入了自动软件校时,但由于硬件本身,每个人做出来的时间误差都不一样,只要修改校时数据就可以了,只要对时准确,天误差一般都可调整到小于一秒。





        下面说一下遥控器的用法,是不是很熟悉啊,电视机顶盒的遥控器......
        电源键:进入/退出待机状态,待机时第一个小数点闪烁
        待机键:开启/关闭掉电模式
        声道键:切换自动/手动显示
        确认键:手动切换显示界面
        电视键:时间小时/月加一
        点播键:时间小时/月减一
        广播键:时间分钟/年/日加一
        资讯键:时间分钟/年/日减一
        调整方法:在各自的显示界面,按加一或减一键即可调整时间/年份/日月



        制作遥控单片机数码管时钟的注意事项:
        1、先说一下数码管断码的问题,大家可以看到,一共用了两个二位一体的数码管,第一个是正放的,第二个是反放的,所以两个数码管的引脚顺序是不一样的,再有就是根据硬件来接线的,为了方便走线,所以数码管的段码引脚和P0口的连接不是按照a,b,c,d,e,f,g,dp的顺序来的,所以大家可能会觉得程序里的数码管段码数组会很奇怪,只要大家自己根据自己的接法改一下就好了。
        2、硬件布局方面,由于都是没有钱的学生,那种专门的印制的电路板我们是做不起的,但是大家都应该有洞洞板(万用板)。洞洞板是个好东西,在洞洞板上做东西能极大的提高我们硬件布局和接线的能力,要尽量少使用飞线,元器件的摆放要合理,比如说晶振与芯片的连接不能太远,18B20温度传感器要远离单片机,数码管等会发热的元器件,不然就不准了......
        3、软件方面,理论上说,软件是万能的,但有时候一个复杂的软件所完成的工作只需几个简单的硬件就能完成,我们在考虑成本的基础上最好还是选择硬件,毕竟现在很多的电子元件都很便宜,因为复杂的软件会增加CPU的负担且降低系统工作的稳定性。当然,反之亦然......
        4、关于DS1302发热的问题,可能很多朋友都遇到过DS1302发热的问题,首先请查看你的DS1302是不是1108批次的货(芯片上面有),这个批次的货有缺陷,一般情况下换几个1302会好的。再有试着将VCC1和VCC2调换这接一下试试。如果再不行,请检查程序。
        5、由于程序及电路图这里不好贴出来,需要的爱好者可以自己下载
        遥控单片机数码管时钟资料来源于网络,这里得感谢一叶知秋无私提供

源程序:1.头文件DS18B20_3.H:

  1. sbit DQ = P1^0;                // 定义DQ引脚为P2.3

  2. /*******************************  延时函数 ********************************
  3. *  功能:在11.059MHz的晶振条件下调用本函数需要24μs ,然后每次计数需16μ
  4. **************************************************************************/
  5. void DS18_delay(int useconds) {
  6. int s;
  7. for (s=0; s<useconds;s++);
  8. }


  9. /*******************************  复位函数 *******************************
  10. * 功能:完成单总线的复位操作。
  11. * 复位时间为480μs,因此延时时间为(480-24)/16 = 28.5,取29μs
  12. * 经过70μs之后检测存在脉冲,因此延时时间为(70-24)/16 = 2.875,取3μs
  13. **************************************************************************/
  14. unsigned char ow_reset(void) {
  15. unsigned char presence;
  16. DQ = 0;                          // 将 DQ 线拉低
  17. DS18_delay(29);                         // 保持 480μs
  18. DQ = 1;                         // DQ返回高电平
  19. DS18_delay(3);                         // 等待存在脉冲
  20. presence = DQ;                 // 获得存在信号
  21. DS18_delay(25);                         // 等待时间隙结束
  22. return(presence);         // 返回存在信号,0 = 器件存在, 1 = 无器件
  23. }


  24. /****************************** 位写入函数 *******************************
  25. * 功能:向单总线写入1位值:bitval
  26. *************************************************************************/
  27. void write_bit(char bitval) {
  28. DQ = 0;                                 // 将DQ 拉低开始写时间隙
  29. if(bitval==1) DQ =1;         // 如果写1,DQ 返回高电平
  30. DS18_delay(5);                                 // 在时间隙内保持电平值,
  31. DQ = 1;         // DS18_delay函数每次循环延时16μs,因此DS18_delay(5) = 104μs
  32. }       


  33. /**************************** 字节写入函数 *******************************
  34. * 功能:向单总线写入一个字节值:val
  35. *************************************************************************/
  36. void ds18write_byte(char val) {
  37. unsigned char i;
  38. unsigned char temp;
  39. for (i=0; i<8; i++) {        // 写入字节, 每次写入一位
  40. temp = val>>i;                
  41. temp &= 0x01;                
  42. write_bit(temp);
  43. }
  44. DS18_delay(5);
  45. }

  46. /**************************** 位读取函数 ********************************
  47. * 功能:从单总线上读取一位信号,所需延时时间为15μs,因此无法调用前面定义
  48. * 的DS18_delay()函数,而采用一个for()循环来实现延时。
  49. * ***********************************************************************/
  50. unsigned char read_bit(void) {
  51. unsigned char i;
  52. DQ = 0;                      //将DQ 拉低开始读时间隙
  53. DQ = 1;                         // then return high
  54. for (i=0; i<3; i++);         // 延时15μs
  55. return(DQ);                         // 返回 DQ 线上的电平值
  56. }

  57. /**************************** 字节读取函数 *******************************
  58. * 功能:从单总线读取一个字节的值
  59. *************************************************************************/
  60. unsigned char DSread_byte(void) {
  61. unsigned char i;
  62. unsigned char value = 0;
  63. for (i=0;i<8;i++) {                                  // 读取字节,每次读取一个字节
  64. if(read_bit()) value|=0x01<<i;         // 然后将其左移
  65. DS18_delay(6);                                        
  66. }
  67. return(value);
  68. }


  69. /******************************* 读取温度函数 *****************************
  70. * 功能:如果单总线节点上只有一个器件则可以直接掉用本函数。如果节点上有多个器
  71. *      件,为了避免数据冲突,应使用Match ROM函数来选中特定器件。
  72. * 注: 本函数是根据DS1820的温度数据格式编写的,若用于DS18B20,必须根据
  73. *      DS18B20的温度数据格式作适当修改。
  74. **************************************************************************/unsigned
  75. int ReadTemperature(void) {
  76. unsigned char get[10];
  77. unsigned char temp_lsb,temp_msb;
  78. unsigned int t;
  79. unsigned char k;
  80. ow_reset();
  81. ds18write_byte(0xCC);                                         // 跳过 ROM
  82. ds18write_byte(0x44);                                         // 启动温度转换
  83. DS18_delay(5);
  84. ow_reset();
  85. ds18write_byte(0xCC);                                         // 跳过 ROM
  86. ds18write_byte(0xBE);                                         // 读暂存器
  87. for (k=0;k<2;k++){get[k]=DSread_byte();}

  88. temp_msb = get[1]; // Sign byte + lsbit
  89. temp_lsb = get[0]; // Temp data plus lsb

  90. t=temp_msb*256+temp_lsb;
  91. t=t&0x0ff0;
  92. if(t<0xff&&t>0xf0)
  93. t=(-1)*t;
  94. return t>>4;
  95. //temp_f = (((int)temp_c)* 9)/5 + 32;
  96.                  // 输出华氏温度值

  97. }
复制代码

2.主程序:
  1. /**********************************WAYNE*********************************/
  2. #include<reg52.h>
  3. #include<intrins.h>
  4. #include"DS18B20_3.H"       //18B20温度传感器头文件,18B20的服务程序都在这里
  5. #define uchar unsigned char
  6. #define uint  unsigned int
  7. unsigned int distemp=8;       //红外接收返回值
  8. unsigned char IRCOM[7];
  9. unsigned char temp;
  10. unsigned char sec, min, hour, day, month, year;
  11. unsigned char aa=0;       //遥控器中间变量
  12. unsigned char bb=1;       //显示界面中间变量
  13. unsigned char cc=0;       //自动切换中间变量
  14. unsigned char dd=0;       //掉电模式中间变量
  15. unsigned char ee=0;
  16. unsigned char flag;
  17. uchar code table[10] = {0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};       //数码管段码
  18. sbit ACC0 = ACC^0;
  19. sbit ACC7 = ACC^7;
  20. sbit T_CLK = P3^5;       //实时时钟时钟线引脚
  21. sbit T_IO = P3^6;       //实时时钟数据线引脚
  22. sbit T_RST = P3^7;       //实时时钟复位线引脚
  23. sbit Y1=P2^6;       //第一位数码管共阳极引脚
  24. sbit Y2=P2^3;       //第二位数码管共阳极引脚
  25. sbit Y3=P2^1;       //第三位数码管共阳极引脚
  26. sbit Y4=P2^0;       //第四位数码管共阳极引脚
  27. sbit led=P0^7;       //数码管小数点阴极引脚
  28. sbit IRIN = P3^2;       //红外接收器数据线与单片机引脚之间的定义

  29. /*********************************延时函数**************************************/
  30. void Delay_1ms(uint i)       //1ms延时
  31. {
  32.         uchar x,j;
  33.         for(j=0;j<i;j++)
  34.         for(x=0;x<=148;x++);       
  35. }
  36. void delay1(unsigned char x)       //延时函数x*0.14MS
  37. {
  38. unsigned char i;
  39.   while(x--)
  40. {
  41.   for (i = 0; i<13; i++) {}
  42. }
  43. }

  44. /**********************************往DS1302写入1Byte数据*************************************/
  45. void v_RTInputByte(uchar ucDa)
  46. {
  47.         uchar i;
  48.         ACC = ucDa;
  49.         T_RST = 1;
  50.         for(i=8; i>0; i--)
  51.         {
  52.                 T_IO = ACC0;
  53.                 T_CLK = 1;
  54.                 T_CLK = 0;
  55.                 ACC = ACC >> 1;
  56.         }
  57. }

  58. /******************************从DS1302读取1Byte数据*****************************************/
  59. uchar uc_RTOutputByte(void)
  60. {
  61.         uchar i;
  62.         T_RST = 1;
  63.         for(i=8; i>0; i--)
  64.         {
  65.                 ACC = ACC >>1;
  66.                 T_IO=1;
  67.                 ACC7 = T_IO;
  68.                 T_CLK = 1;
  69.                 T_CLK = 0;
  70.         }
  71.         return(ACC);
  72. }

  73. /*******************************往DS1302写入数据****************************************/
  74. void v_W1302(uchar ucAddr, uchar ucDa)
  75. {
  76.         T_RST = 0;
  77.         T_CLK = 0;
  78.         T_RST = 1;
  79.         v_RTInputByte(ucAddr);       // 写地址
  80.         _nop_();
  81.         _nop_();
  82.         v_RTInputByte(ucDa);       // 写1Byte数据
  83.         T_CLK = 1;
  84.         T_RST = 0;
  85. }

  86. /**********************************读取DS1302某地址的数据*************************************/
  87. uchar uc_R1302(uchar ucAddr)
  88. {
  89.         uchar ucDa;
  90.         T_RST = 0;
  91.         T_CLK = 0;
  92.         T_RST = 1;
  93.         v_RTInputByte(ucAddr);       //写地址,命令
  94.         _nop_();
  95.         _nop_();
  96.         ucDa = uc_RTOutputByte();       //读1Byte数据
  97.         T_CLK = 1;
  98.         T_RST = 0;
  99.         return(ucDa);
  100. }

  101. /*********************************DEC码转换为BCD码**************************************/
  102. uchar dectobcd(uchar dec)
  103. {
  104.         uchar bcd;
  105.         bcd = 0;
  106.         while(dec >= 10)
  107.         {              
  108.                 dec -= 10;                        
  109.                 bcd++;
  110.         }
  111.         bcd <<= 4;
  112.         bcd |= dec;
  113.         return bcd;
  114. }

  115. /*******************************BCD码转换为DEC码****************************************/
  116. uchar bcdtodec(uchar bcd)
  117. {
  118.         uchar data1;
  119.         data1 = bcd & 0x0f;       //取BCD低4位
  120.         bcd = bcd & 0x70;       //剔除BCD的最高位和低4位。
  121.         data1 += bcd >> 1;
  122.         data1 += bcd >> 3;       //用位移代替乘法运算
  123.         return data1;
  124. }

  125. /*********************************往DS1302中写入数据**************************************/
  126. void Write_DS1302Init(void)
  127. {
  128.         v_W1302(0x8e,0x00);       //屏蔽一下语句,则掉电走时有效;屏蔽则掉电走时失效,且每次上电后显示12:00(时)/2012(年)/1(月)1(日)
  129.         v_W1302(0x80,0x00);       //写入秒
  130.         v_W1302(0x82,0x00);       //写入分
  131.         v_W1302(0x84,0x12);       //写入小时
  132.         v_W1302(0x86,0x01);       //写入日
  133.         v_W1302(0x88,0x01);       //写入月
  134.         v_W1302(0x8a,0x00);       //写入星期
  135.         v_W1302(0x8c,0x12);       //写入年
  136.         v_W1302(0x8e,0x80);
  137. }

  138. /*********************************数码管位选码**************************************/
  139. void Y_1()       //选通第一位数码管
  140. {
  141.   Y1=0;
  142.   Y2=1;
  143.   Y3=1;
  144.   Y4=1;
  145. }
  146. void Y_2()       //选通第二位数码管
  147. {
  148.   Y1=1;
  149.   Y2=0;
  150.   Y3=1;
  151.   Y4=1;
  152. }
  153. void Y_3()       //选通第三位数码管
  154. {
  155.   Y1=1;
  156.   Y2=1;
  157.   Y3=0;
  158.   Y4=1;
  159. }
  160. void Y_4()       //选通第四位数码管
  161. {
  162.   Y1=1;
  163.   Y2=1;
  164.   Y3=1;
  165.   Y4=0;
  166. }
  167. void Y_5()       //切断所有数码管,用于数码管消隐
  168. {
  169.   Y1=1;
  170.   Y2=1;
  171.   Y3=1;
  172.   Y4=1;
  173. }


  174. /*******************************显示程序****************************************/
  175. void Run_DS1302(void)
  176. {
  177.                 v_W1302(0x8f, 0x00);
  178.                 sec = bcdtodec(uc_R1302(0x81));       //读出DS1302中的秒
  179.                 min = bcdtodec(uc_R1302(0x83));       //读出DS1302中的分
  180.                 hour = bcdtodec(uc_R1302(0x85));       //读出DS1302中的小时
  181.                 day = bcdtodec(uc_R1302(0x87));       //读出DS1302中的日
  182.                 month = bcdtodec(uc_R1302(0x89));       //读出DS1302中的月
  183.                 year = bcdtodec(uc_R1302(0x8d));       //读出DS1302中的年   
  184.             v_W1302(0x8f,0x80);
  185.                 if(distemp == 0)
  186.                 {   distemp = 8;
  187.                     aa++;
  188.                 }
  189.                 if(distemp ==7)
  190.                 {   distemp = 8;
  191.                     ee++;
  192.                 }
  193.                 if(ee == 2)
  194.                 ee=0;
  195.                 if(aa==4)       //判断显示数据
  196.                 aa=0;
  197.                 if(aa == 0)
  198.                 {
  199.                         bb = 1;       //显示时间
  200.                 }
  201.                 if(aa == 1)
  202.                 {
  203.                         bb = 2;       //显示温度
  204.                 }
  205.         if(aa == 2)
  206.                 {
  207.                         bb = 3;       //显示年份
  208.                 }
  209.                 if(aa == 3)
  210.                 {
  211.                         bb = 4;       //显示日期
  212.                 }
  213.         if(ee == 1)
  214.                 bb = 5;
  215.                 if(bb == 1)       //时间显示界面
  216.                 {   if(hour / 10 % 10!=0)       //十位为0则不显示
  217.                         {P1 = ~table[hour / 10 % 10];       //显示小时的十位数据
  218.              led=1;
  219.                          Y_1();       //选通第一位数码管
  220.                          Delay_1ms(1);       //短暂停留
  221.                          P1 = 0xff;       //数码管消隐
  222.                           led=1;
  223.                          Y_5();       //切断所有数码管
  224.                         }
  225.                         P1 = ~table[hour % 10];       //显示小时的个位数据
  226.                         if(sec%2==0)       //判断是否秒闪烁
  227.                         led=1;
  228.                         else led=0;
  229.                         Y_2();       //选通第二位数码管
  230.                         Delay_1ms(1);       //短暂停留
  231.             P1 = 0xff;       //数码管消隐
  232.                         led=1;
  233.                         Y_5();       //切断所有数码管
  234.                         P1 = ~table[min / 10 % 10];       //显示分钟的十位数据
  235.             if(sec%2==0)       //判断是否秒闪烁
  236.                         led=1;
  237.                         else led=0;
  238.                         Y_3();       //选通第三位数码管
  239.                         Delay_1ms(1);       //短暂停留
  240.                         P1 = 0xff;       //数码管消隐
  241.                         led=1;
  242.                         Y_5();       //切断所有数码管
  243.                         P1 = ~table[min % 10];       //显示分钟的个位数据
  244.             led=1;
  245.                         Y_4();       //选通第四位数码管
  246.                         Delay_1ms(1);       //短暂停留
  247.                         P1 = 0xff;       //数码管消隐
  248.                         led=1;
  249.                         Y_5();       //切断所有数码管
  250.                 }
  251.                 if(bb == 2)       //温度显示界面
  252.                 {
  253.                         P1 = ~table[(flag-1) / 10 % 10];       //显示温度的十位数据
  254.             led=1;
  255.                         Y_2();       //选通第二位数码管
  256.                         Delay_1ms(1);       //短暂停留
  257.                         P1 = 0xff;       //数码管消隐
  258.                         led=1;
  259.                         Y_5();       //切断所有数码管
  260.                         P1 = ~table[(flag-1) % 10];       //显示温度的个位数据
  261.             led=1;
  262.                         Y_3();       //选通第三位数码管
  263.                         Delay_1ms(1);       //短暂停留
  264.                         P1 = 0xff;       //数码管消隐
  265.                         led=1;
  266.                         Y_5();       //切断所有数码管
  267.                         P1 = ~0x58;       //显示摄氏温度标志
  268.                         led=0;
  269.                         Y_4();       //选通第四位数码管
  270.                         Delay_1ms(1);       //短暂停留
  271.                         P1 = 0xff;       //数码管消隐
  272.                         led=1;
  273.                         Y_5();       //切断所有数码管
  274.                 }
  275.                 if(bb == 3)       //年份显示界面
  276.                 {
  277.                         P1 = ~0x6e;       //显示年份千位数据,默认为2
  278.             led=1;
  279.                         Y_1();       //选通第一位数码管
  280.                         Delay_1ms(1);       //短暂停留
  281.                         P1 = 0xff;       //数码管消隐
  282.                         led=1;
  283.                         Y_5();       //切断所有数码管
  284.                         P1 = ~0x77;       //显示年份百位数据,默认为0
  285.             led=1;
  286.                         Y_2();       //选通第二位数码管
  287.                         Delay_1ms(1);       //短暂停留
  288.                         P1 = 0xff;       //数码管消隐
  289.                         led=1;
  290.                         Y_5();       //切断所有数码管
  291.                     P1 = ~table[year / 10 % 10];       //显示年份十位数据
  292.             led=1;
  293.                         Y_3();       //选通第三位数码管
  294.                         Delay_1ms(1);       //短暂停留
  295.                         P1 = 0xff;       //数码管消隐
  296.                         led=1;
  297.                         Y_5();       //切断所有数码管
  298.                         P1 = ~table[year % 10];       //显示年份个位数据
  299.             led=1;
  300.                         Y_4();       //选通第四位数码管
  301.                         Delay_1ms(1);       //短暂停留
  302.                         P1 = 0xff;       //数码管消隐
  303.                         led=1;
  304.                         Y_5();       //切断所有数码管
  305.                 }
  306.                 if(bb == 4)       //月日显示界面
  307.                 {   if(month / 10 % 10!=0)       //十位为0则不显示
  308.                         {P1 = ~table[month / 10 % 10];       //显示月十位数据
  309.              led=1;
  310.                          Y_1();       //选通第一位数码管
  311.                          Delay_1ms(1);       //短暂停留
  312.              P1 = 0xff;       //数码管消隐
  313.                            led=1;
  314.                          Y_5();       //切断所有数码管
  315.                          }
  316.                         P1 = ~table[month % 10];       //显示月个位数据
  317.             led=0;
  318.                         Y_2();       //选通第二位数码管
  319.                         Delay_1ms(1);       //短暂停留
  320.                         P1 = 0xff;       //数码管消隐
  321.                         led=1;
  322.                         Y_5();       //切断所有数码管                       
  323.                         if(day / 10 % 10!=0)       //十位为0则不显示
  324.                         {P1 = ~table[day / 10 % 10];       //显示日十位数据
  325.              led=1;
  326.                          Y_3();       //选通第三位数码管
  327.                          Delay_1ms(1);       //短暂停留
  328.                          P1 = 0xff;       //数码管消隐
  329.                          led=1;
  330.                          Y_5();       //切断所有数码管
  331.                          }
  332.                         P1 = ~table[day % 10];       //显示日个位数据
  333.             led=1;
  334.                         Y_4();       //选通第四位数码管
  335.                         Delay_1ms(1);       //短暂停留
  336.                         P1 = 0xff;       //数码管消隐
  337.                         led=1;
  338.                         Y_5();       //切断所有数码管                       
  339.                 }
  340.                 if(bb == 5)
  341.                 {  
  342.                  P1=0xff;
  343.                  if(sec%2==0)       //判断是否秒闪烁
  344.                         led=1;
  345.                         else led=0;
  346.          Y_1();       //选通第四位数码管
  347.                  Delay_1ms(1);       //短暂停留
  348.                  }
  349. }

  350. /**********************************自动切换程序*************************************/
  351. void zdqh()
  352. {
  353. if(distemp==6)
  354. {distemp=8;
  355.   cc++;
  356. }
  357. if(cc==2)       //切换为手动显示
  358. cc=0;
  359. if(cc==1)       //切换为自动循环显示
  360. {
  361.   if(sec%10==0)       //每10秒内,前7秒显示时间界面
  362.   aa = 0;
  363.   if(sec==7||sec==37)       //7秒和37秒切换为温度显示
  364.   aa = 1;
  365.   if(sec==17||sec==47)       //17秒和47秒切换为年份显示
  366.   aa = 2;
  367.   if(sec==27||sec==57)       //27秒和57秒切换为月日显示
  368.   aa = 3;
  369. }
  370. }

  371. /*********************************调节程序**************************************/
  372. void tj()
  373. {
  374. if(distemp == 1)       //判断按下的键值
  375. {distemp=8;       //返回值回位
  376.   if(bb==1)       //判断当前显示界面
  377.   {
  378.    hour++;       //小时加一调节
  379.    if(hour==24)       //限制小时数据在0~23之间
  380.    hour=0;
  381.    temp=dectobcd(hour);       //将十进制数转换成十六禁止数
  382.    v_W1302(0x8e,0x00);       //禁止写保护
  383.    v_W1302(0x84,temp);       //写入小时数据
  384.    v_W1302(0x80,0x00);
  385.    v_W1302(0x8e,0x80);       //打开写保护
  386.   }
  387.   if(bb==4)       //判断当前显示界面
  388.   {
  389.    month++;       //月加一调节
  390.    if(month==13)       //限制月数据在1~12之间
  391.    month=1;
  392.    temp=dectobcd(month);       //将十进制数转换成十六禁止数
  393.    v_W1302(0x8e,0x00);       //禁止写保护
  394.    v_W1302(0x88,temp);       //写入月数据
  395.    v_W1302(0x8e,0x80);       //打开写保护
  396.   }
  397. }
  398. if(distemp == 2)       //判断按下的键值
  399. {distemp=8;       //返回值回位
  400.   if(bb==1)       //判断当前显示界面
  401.   {
  402.    min++;//分钟加一调节
  403.    if(min==60)//限制分钟数据在0~59之间
  404.    min=0;
  405.    temp=dectobcd(min);//将十进制数转换成十六禁止数
  406.    v_W1302(0x8e,0x00);//禁止写保护
  407.    v_W1302(0x82,temp);//写入分钟数据
  408.    v_W1302(0x80,0x00);
  409.    v_W1302(0x8e,0x80);//打开写保护
  410.   }
  411.   if(bb==3)//判断当前显示界面
  412.   {
  413.    year++;//年份加一调节
  414.    if(year==100)//限制年份数据在0~99之间
  415.    year=0;
  416.    temp=dectobcd(year);//将十进制数转换成十六禁止数
  417.    v_W1302(0x8e,0x00);//禁止写保护
  418.    v_W1302(0x8c,temp);//写入年数据
  419.    v_W1302(0x8e,0x80);//打开写保护
  420.   }
  421.   if(bb==4)//判断当前显示界面
  422.   {
  423.    day++;       //日加一调节
  424.    if(day==32)       //限制日数据在1~31之间
  425.    day=1;
  426.    temp=dectobcd(day);       //将十进制数转换成十六禁止数
  427.    v_W1302(0x8e,0x00);       //禁止写保护
  428.    v_W1302(0x86,temp);       //写入日数据
  429.    v_W1302(0x8e,0x80);       //打开写保护
  430.   }
  431. }
  432. if(distemp == 3)       //判断按下的键值
  433. {distemp=8;       //返回值回位
  434.   if(bb==1)       //判断当前显示界面
  435.   {
  436.    hour--;       //小时减一调节
  437.    if(hour==-1)       //限制小时数据在0~23之间
  438.    hour=23;
  439.    temp=dectobcd(hour);       //将十进制数转换成十六禁止数
  440.    v_W1302(0x8e,0x00);       //禁止写保护
  441.    v_W1302(0x84,temp);       //写入小时数据
  442.    v_W1302(0x80,0x00);
  443.    v_W1302(0x8e,0x80);       //打开写保护
  444.   }
  445.   if(bb==4)       //判断当前显示界面
  446.   {
  447.    month--;       //月减一调节
  448.    if(month==0)       //限制月数据在1~12之间
  449.    month=12;
  450.    temp=dectobcd(month);       //将十进制数转换成十六禁止数
  451.    v_W1302(0x8e,0x00);       //禁止写保护
  452.    v_W1302(0x88,temp);       //写入月数据
  453.    v_W1302(0x8e,0x80);       //打开写保护
  454.   }
  455. }
  456. if(distemp == 4)       //判断按下的键值
  457. {distemp=8;       //返回值回位
  458.   if(bb==1)       //判断当前显示界面
  459.   {
  460.    min--;       //分钟减一调节
  461.    if(min==-1)       //限制分钟数据在0~59之间
  462.    min=59;
  463.    temp=dectobcd(min);       //将十进制数转换成十六禁止数
  464.    v_W1302(0x8e,0x00);       //禁止写保护
  465.    v_W1302(0x82,temp);       //写入分钟数据
  466.    v_W1302(0x80,0x00);
  467.    v_W1302(0x8e,0x80);       //打开写保护
  468.   }
  469.   if(bb==3)       //判断当前显示界面
  470.   {
  471.    year--;       //年份减一调节
  472.    if(year==-1)       //限制年份数据在0~99之间
  473.    year=99;
  474.    temp=dectobcd(year);       //将十进制数转换成十六禁止数
  475.    v_W1302(0x8e,0x00);       //禁止写保护
  476.    v_W1302(0x8c,temp);       //写入年数据
  477.    v_W1302(0x8e,0x80);       //打开写保护
  478.   }
  479.   if(bb==4)       //判断当前显示界面
  480.   {
  481.    day--;       //日减一调节
  482.    if(day==0)       //限制日数据在1~31之间
  483.    day=31;
  484.    temp=dectobcd(day);       //将十进制数转换成十六禁止数
  485.    v_W1302(0x8e,0x00);       //禁止写保护
  486.    v_W1302(0x86,temp);       //写入日数据
  487.    v_W1302(0x8e,0x80);       //打开写保护
  488.   }
  489. }
  490. }

  491. /*******************************掉电模式程序****************************************/
  492. void xmms()
  493. {
  494. if(dd == 1)       //单片机进入掉电工作模式,该模式下只有首位数码管的小数点常亮
  495. {
  496.   P0=~0xff;       //在掉电模式下,只有第一位数码管的小数点亮
  497.   led=1;
  498.   Y_5();
  499.   aa=0;
  500.   ee=0;
  501.   PCON=0x02;       //设置掉电模式寄存器,这句一定要放在该子程序的最后
  502. }
  503. }

  504. /********************************自动校时***************************************/
  505. void zdjs()
  506. {
  507. if(hour==3&&min==0&&sec==30)
  508.    {
  509.         v_W1302(0x8e,0x00);      
  510.         v_W1302(0x80,0x00);
  511.     v_W1302(0x8e,0x80);
  512.     }
  513. }

  514. /********************************主函数***************************************/
  515. void Main(void)
  516. {   
  517.     EA=1;       //打开总中断
  518.         EX0=1;       //允许外部0中断
  519.     IT0=1;       //设置外部中断0为边沿方式触发
  520.     IRIN=1;
  521.         Write_DS1302Init();
  522.         while(1)       //无限循环以下子程序
  523.         {
  524.          Run_DS1302();       //显示程序
  525.          flag=ReadTemperature();       //将18b2头文件运行返回的函数结果送到变量flag中,用于显示
  526.          tj();       //调节程序
  527.          xmms();       //休眠模式
  528.          zdqh();       //自动切换
  529.          zdjs();       //自动校时
  530.          }
  531. }

  532. /*********************红外接收程序**********************/
  533. void IR_IN() interrupt 0       //外部中断0
  534. {
  535.   unsigned char p,q,R=0;
  536.   EA = 0;
  537.   delay1(15);
  538.   if (IRIN==1)
  539.    { EA =1;
  540.          return;
  541.         }                  
  542.   while (!IRIN)            
  543.     {delay1(1);}
  544.   for (p=0;p<4;p++)         
  545.   {
  546.    for (q=0;q<8;q++)      
  547.    {
  548.     while (IRIN)            
  549.      {delay1(1);}
  550.     while (!IRIN)        
  551.      {delay1(1);}
  552.     while (IRIN)         
  553.      {
  554.       delay1(1);
  555.       R++;           
  556.       if (R>=30)
  557.            { EA=1;
  558.              return;}               
  559.       }                                 
  560.      IRCOM[p]=IRCOM[p] >> 1;                 
  561.      if (R>=8) {IRCOM[p] = IRCOM[p] | 0x80;}
  562.      R=0;
  563.   }                        
  564. }                          
  565.    if (IRCOM[2]!=~IRCOM[3])
  566.    { EA=1;
  567.      return; }
  568.    switch(IRCOM[2])
  569.    {
  570.                    case 0x07:       //显示切换键
  571.                         distemp = 0;
  572.                         break;
  573.                 case 0x44:       //左加一键
  574.                         distemp = 1;
  575.                         break;
  576.                 case 0x45:       //右加一键
  577.                         distemp = 2;
  578.                         break;
  579.                 case 0x46:       //左减一键
  580.                         distemp = 3;
  581.                         break;
  582.                 case 0x47:       //右减一键
  583.                         distemp = 4;
  584.                         break;
  585.         case 0x16:       //掉电模式开关键
  586.                         distemp = 5;
  587.                         break;
  588.                 case 0x10:       //自动切换开关键
  589.                         distemp = 6;
  590.                         break;
  591.                 case 0x1c:       //待机模式开关键
  592.                         distemp = 7;
  593.                         break;
  594.    }
  595.     if(distemp == 5)       //开启、关闭掉电模式,因掉电模式只能由外部中断唤醒,所以下面的程序必须放在中断函数中
  596.     {distemp = 8;       //返回值回位
  597.      dd++;
  598.     }
  599.     if(dd == 2)
  600.     dd = 0;
  601.         if(dd == 0)       //退出掉电模式
  602.         {PCON=0x00;       //设置掉电模式工作寄存器
  603.          }
  604.         if(dd == 1)       //掉电模式下,其他按键无效
  605.         distemp = 8;       //不返回任何键值
  606.         if(ee==1)
  607.          {if(distemp==0||distemp==1||distemp==2||distemp==3||distemp==4||distemp==6)
  608.           distemp = 8;
  609.           }
  610.         EA = 1;       //打开总中断
  611. }
复制代码




分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏14 分享淘帖 顶 踩1
回复

使用道具 举报

沙发
ID:89183 发表于 2015-9-25 22:53 | 只看该作者
很好谢谢楼主分享,有没有带闹钟版的啊,有的话给我一份吧谢谢了 xstao@qq.com
回复

使用道具 举报

板凳
ID:95498 发表于 2015-11-12 11:54 | 只看该作者
感谢楼主!!!!很有用
回复

使用道具 举报

地板
ID:95976 发表于 2015-11-16 14:21 | 只看该作者
感谢楼主分享
回复

使用道具 举报

5#
ID:33101 发表于 2015-12-10 10:40 | 只看该作者
感谢楼主分享
回复

使用道具 举报

6#
ID:92539 发表于 2015-12-21 08:24 | 只看该作者
很好,正有急用
回复

使用道具 举报

7#
ID:41337 发表于 2015-12-23 23:09 | 只看该作者
感谢楼主分享
回复

使用道具 举报

8#
ID:64089 发表于 2016-2-17 13:52 | 只看该作者
不错哦,我始终没有搞懂红外遥控后如何退出
回复

使用道具 举报

9#
ID:105697 发表于 2016-2-17 22:09 来自手机 | 只看该作者
酷!!!谢谢楼主
回复

使用道具 举报

10#
ID:120731 发表于 2016-5-17 17:59 | 只看该作者
看一下,向大神学习
回复

使用道具 举报

11#
ID:126447 发表于 2016-6-15 21:34 | 只看该作者
资料不错值得一看
回复

使用道具 举报

12#
ID:126485 发表于 2016-6-15 23:24 | 只看该作者
谢谢楼主学习了
回复

使用道具 举报

13#
ID:126935 发表于 2016-6-16 09:11 | 只看该作者
谢谢楼主
回复

使用道具 举报

14#
ID:145554 发表于 2016-11-1 21:07 | 只看该作者
很好的资料要看看
回复

使用道具 举报

15#
ID:93911 发表于 2016-11-15 21:40 | 只看该作者
值得学习。谢谢分享
回复

使用道具 举报

16#
ID:146910 发表于 2016-11-26 19:57 | 只看该作者
很好的资料要看看
回复

使用道具 举报

17#
ID:146910 发表于 2016-11-27 19:28 | 只看该作者
        v_W1302(0x8e,0x00);       //屏蔽一下语句,则掉电走时有效;屏蔽则掉电走时失效,且每次上电后显示12:00(时)/2012(年)/1(月)1(日)
        v_W1302(0x80,0x00);       //写入秒
        v_W1302(0x82,0x00);       //写入分
        v_W1302(0x84,0x12);       //写入小时
        v_W1302(0x86,0x01);       //写入日
        v_W1302(0x88,0x01);       //写入月
        v_W1302(0x8a,0x00);       //写入星期
        v_W1302(0x8c,0x12);       //写入年
        v_W1302(0x8e,0x80);
要怎么修改下电后不初使化(每次上电后显示12:00(时)/2012(年)/1(月)1(日)
回复

使用道具 举报

18#
ID:157847 发表于 2016-12-29 18:37 | 只看该作者
挺好的
回复

使用道具 举报

19#
ID:136460 发表于 2016-12-31 10:47 | 只看该作者
很好的资料要看看
回复

使用道具 举报

20#
ID:35873 发表于 2017-6-19 01:55 | 只看该作者
这个详细  简直就是一教程 支持下
回复

使用道具 举报

21#
ID:159139 发表于 2017-6-20 23:50 | 只看该作者
谢谢分享,我下了一个,学学分时显示。
回复

使用道具 举报

22#
ID:270854 发表于 2018-1-21 10:58 | 只看该作者
不显示
回复

使用道具 举报

23#
ID:314979 发表于 2018-4-24 14:49 | 只看该作者
我下了一个,学学
回复

使用道具 举报

24#
ID:315554 发表于 2018-5-3 17:32 | 只看该作者
楼主好,很喜欢这个程序,选了同样的MCU晶振等配件,搭起硬件,写入程序,只显示两个秒点(正常秒闪),数字不显示,请教什么原因?
回复

使用道具 举报

25#
ID:492589 发表于 2019-3-17 21:58 | 只看该作者
有没有图
回复

使用道具 举报

26#
ID:492822 发表于 2019-4-6 14:17 | 只看该作者
楼主能不能发一下原理图啊!急用啊
回复

使用道具 举报

27#
ID:492822 发表于 2019-4-7 18:52 | 只看该作者
nsj21n 发表于 2016-2-17 13:52
不错哦,我始终没有搞懂红外遥控后如何退出

求原理图怎么画
回复

使用道具 举报

28#
ID:168038 发表于 2019-4-8 17:47 | 只看该作者
资料不错,做一回伸手党了,拿来就用
回复

使用道具 举报

29#
ID:21568 发表于 2019-6-12 14:58 | 只看该作者
程序跟电路板的图对不上啊,是P1口还是P0口?
回复

使用道具 举报

30#
ID:759683 发表于 2020-7-5 19:14 | 只看该作者
没有原理图,经试验程序和电路不符
回复

使用道具 举报

31#
ID:517951 发表于 2020-11-15 17:31 | 只看该作者
很详细不错,值得学习。51hei有你更精彩!!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表