找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5827|回复: 1
收起左侧

可设置8个闹钟时间的单片机智能时钟

[复制链接]
ID:104835 发表于 2016-2-2 04:38 | 显示全部楼层 |阅读模式
                        普通的家用时钟一般只能设置一个闹钟时间,但很多人均需为工作日、周末、早晨、午休等不同时段设置不同的闹钟时间。如果是使用普通的闹钟,只好每次休息前重新设置,很不方便,有时甚至会出现忘记更改闹钟设置而睡过头的情况。
        
        
        针对这种情况,本人利用89C51单片机设计了一款有8种闹钟设置的时钟,通过一段时间的使用,情况良好。
        
        
        1、                          元件清单
        
        共阴极数码管                                              8个                                              4511七段译码芯片                                        1片
        
        ATMEL89C51单片机                                          1片                                              24C08EEPROM                     1片
        
        24M晶振                                                   1个                                              9V变压器(3VA)                                         1个
        
        LM317输出可调稳压IC                                       1个                                                      整流桥堆                                                1个
        
        470uF电容                                                 1个                                              100uF电容                                               1个
        
        10uF电容                                                  1个                                              0.1uF电容                                               1个
        
        33pF电容                                                  2个                                                      蜂鸣器                                                  1个
        
        9014(或其它NPN管)                                       8个                                              ksp92(或其它PNP管)            1个
        
        二极管                                                    2个                                              1K欧电阻                                                16个
        
        470欧电阻                                                 1个                                              10K欧可调电阻                                           1个
        
        10K欧电阻                                                 4个                                                      按钮开关                                                4个        
        
        可装四节电池的电池盒                                      1个                                                      万能板(约12CM*17CM)           1块
        
        所有元件按以下的电路图焊接在一块万能板上。注意LM317的输出应由低调高,以免烧IC。
        
        
        1、                          电路图
        
         image002.jpg
        
        
        1、                          功能简介
        
        该时钟以24小时制显示时间,并可显示2000年至2049年之间的任何日期及星期,日期与时间经按键可相互切换,可输入8个闹钟时间设置,每个闹钟设置包括响铃的时间(小时与分钟)、对工作日有效还是对周末有效的标志,以及本项设置是否启用的标志等三部分。这8个闹钟设置均保存在EEPROM中,即使掉电也不用重新输入。当然使用者可通过按钮对任何一个设置作修改。数码管可经按钮关闭显示,避免夜间刺眼、影响睡眠。调节LM317输出电压,可改变数码管亮度,但电压不能低于后备电池的电压,否则后备电池供电。用四节1.5V电池串联作后备电源,保证市电停电时时钟继续走时。时钟的精度取决于晶振频率的精度。
        
        
        2、                          程序清单
        
        本程序用C语言编写,经Keil                         C51编译成二进制码后写入89C51内的EPROM内即可。
  1. #include "atmelat89x51.h"

  2. #include "intrins.h"



  3. unsigned char hour,min,sec,year,month,day,weekday;   //当前时间、日期、星期

  4. unsigned int  count_down;      //1秒钟计时用



  5. bit led_on;                    //数码管是否点亮的标志

  6. unsigned char display[8];         //8位数码管要显示的数据

  7. unsigned char attr;              //八个数码管的闪烁控制字节,当为0时,对应数码管闪

  8. bit flash;                            //LED 闪烁开关,与attr共同决定数码管是否闪烁

  9. unsigned char show_status;     //LED 显示状态标志

  10.         // 0:设置闹钟数据

  11. // 1:显示当前日期及星期

  12.                                                  // 2:显示当前时间

  13. // 3:设置当前日期

  14. // 4:设置当前时间



  15. bit km;                      //按键已去抖动标志

  16. bit kp;                       //按键已处理标志



  17. bit sound;                    //蜂鸣器响标志

  18. bit alarm_stop;                //蜂鸣器响后用户手工按停标志

  19. struct  { unsigned char h;           //小时

  20.         unsigned char m;           //分钟

  21.        } alarm[8];                            //8个闹钟项

  22. unsigned char alarm_en;        //闹钟项启用标志

  23. unsigned char alarm_wk;        //闹钟项周末启用标志

  24. unsigned char cur_alarm_set;     //当前设置的闹钟项

  25. unsigned char cur_alarm_active;   //当前到点的闹钟项

  26. bit new_alarm_info;            //闹钟项内容已修改标志



  27. sbit sound_output = P1^5;      //蜂鸣器驱动端口,输出0时蜂鸣器响

  28. sbit SDA_PIN   = P1^6;          //EEPROM数据线端口

  29. sbit SCL_PIN   = P1^7;          //EEPROM时钟线端口



  30. void I2cDelay()               //EEPROM操作时需要的延时函数

  31. { _nop_();  _nop_();  _nop_();

  32.   _nop_();  _nop_();  _nop_();

  33. }

  34. void DelayX1ms(unsigned char count)        //延迟函数,参数为毫秒数

  35. {unsigned char i,j;

  36.   for(i=0;i<count;i++)

  37.     for(j=0;j<240;j++)    ;

  38. }

  39. void Start()                                        //I2C启动,24C08使用I2C方式

  40. { SDA_PIN=1;   I2cDelay();

  41.   SCL_PIN=1;   I2cDelay();

  42.   SDA_PIN=0;   I2cDelay();

  43.   SCL_PIN=0;   

  44. }

  45. void Stop()                         //I2C停止

  46. { I2cDelay();  SDA_PIN=0;

  47.   I2cDelay();  SCL_PIN=1;

  48.   I2cDelay();  SDA_PIN=1;

  49.   I2cDelay();

  50. }

  51. bit SendByte(unsigned char value)               //发送1字节数据给EEPROM

  52. {unsigned char i;

  53. bit no_ack=0;

  54.   for(i=0;i<8;i++)                     //发送8位数据

  55.   { I2cDelay();   

  56. if(value&0x80)   SDA_PIN=1;

  57. else            SDA_PIN=0;

  58.     value=value<<1;        

  59.        I2cDelay();      SCL_PIN=1;         

  60.        I2cDelay();

  61.        I2cDelay();      SCL_PIN=0;

  62.    }

  63.   I2cDelay();        SDA_PIN=1;     //确认脉冲周期,等待EEPROM的确认

  64.   I2cDelay();        SCL_PIN=1;      

  65.   I2cDelay();      

  66.   if(SDA_PIN==1)   no_ack=1;

  67.   I2cDelay();        SCL_PIN=0;

  68.   return no_ack;

  69. }

  70. void mywrite(unsigned char address,unsigned char value)         //向EEPROM写1字节

  71. { Start();          SendByte(0xa0);     SendByte(address);   

  72.   SendByte(value);  Stop();             DelayX1ms(10);

  73. }

  74. unsigned char ReadByte()                    //从EEPROM接收1字节

  75. {unsigned char i,bval;

  76.   bval=0;

  77.   for(i=0;i<8;i++)                              //接收8位数据

  78.    { I2cDelay();

  79.      SDA_PIN=1;                          //从P1输入数据时,先往P1输入“1”

  80.      I2cDelay();    SCL_PIN=1;



  81.      I2cDelay();    bval=bval<<1;     if(SDA_PIN)    bval=bval|0x01;

  82.      I2cDelay();    SCL_PIN=0;

  83.     }

  84.        I2cDelay();     SDA_PIN=1;    //确认脉冲周期,不送出确认

  85.        I2cDelay();     SCL_PIN=1;   

  86.        I2cDelay();     

  87.        I2cDelay();



  88.     return(bval);

  89. }

  90. unsigned char myread(unsigned char address)            //从EEPROM读入1字节数据

  91. {unsigned char tmp;

  92.   Start();        SendByte(0xa0);     SendByte(address);   

  93.   Start();        SendByte(0xa1);     tmp=ReadByte();

  94.   Stop();         DelayX1ms(2);

  95.   return(tmp);

  96. }

  97. void Timer0ISR(void) interrupt 1 using 3     //定时器0中断程序,用于走时,1/8000秒一次

  98. {unsigned char tmp,tmp_days;

  99.   count_down--;

  100.   if(count_down==1 || count_down==2001 || count_down==4001 || count_down==6001)

  101.    { flash=~flash;                                   //数码管闪烁的开关量

  102.      if(sound && flash) sound_output=0;         //驱动蜂鸣器

  103.      else             sound_output=1;         //关闭蜂鸣器

  104.         return;

  105.        }

  106.                      /*** 计算当前日期为星期几***/

  107.   if(count_down==3000)

  108.    { if(year==0)       weekday=5;           //2000年1月1日为星期六

  109.     else { tmp=(year-1)/4+1;   tmp=(year-tmp)+tmp*2;

  110.          weekday=(tmp+5)%7;           //计算出当前年的1月1日为星期几

  111.         }

  112.      tmp_days=0;

  113.      for(tmp=1;tmp<month;tmp++)

  114.        if(tmp==1 || tmp==3 || tmp==5 || tmp==7 || tmp==8 || tmp==10)

  115.           tmp_days=tmp_days+31;

  116.        else if(tmp==4 || tmp==6 || tmp==9 || tmp==11)

  117.              tmp_days=tmp_days+30;

  118.        else if(tmp==2)

  119.              { if(year%4==0) tmp_days=tmp_days+29;

  120.             else         tmp_days=tmp_days+28;

  121.            }

  122.      tmp_days=tmp_days+day-1;    weekday=(weekday+tmp_days%7)%7+1;

  123.         return;

  124.        }

  125.                   /***  查询是否有闹钟时间项符合触发条件 ***/

  126.   if(count_down==5000)

  127.   { if((alarm_stop || sound) && alarm[cur_alarm_active].m!=min) //触发后1分钟

  128.       { alarm_stop=0; sound=0; }                                                //自动关蜂鸣器



  129.     if(sound==0 && alarm_stop==0)           //没有已触发的闹钟项

  130.      for(tmp=0;tmp<8;tmp++)                           //则查询8个闹钟项内是否有符合条件的

  131.          { if(((alarm_en>>tmp)&1)==0) continue;    //该闹钟项不启用

  132.            if(((alarm_wk>>tmp)&1)==1)           //该闹钟项周末有效

  133.                 { if(weekday!=6 && weekday!=7) continue;  }     //当前不是星期六或星期天

  134.               else

  135.                 { if(weekday==6 || weekday==7) continue;  }



  136.         if(alarm[tmp].h==hour && alarm[tmp].m==min)             //比较当前时间与该

  137.           { sound=1;  cur_alarm_active=tmp; break; }         //闹钟项的时间

  138.           }

  139.     return;

  140.    }



  141.   if(count_down==0)                              //过了一秒钟

  142.    { count_down=8000;

  143.      sec++;

  144.      if(sec==60)

  145.       { sec=0;

  146.         min++;

  147.         if(min==60)

  148.           { min=0;

  149.             hour++;

  150.             if(hour==24)

  151.               { hour=0;   day++;

  152.                             switch(day)

  153.                             { case 29: if(month==2 && year%4)    { day=1; month=3; }

  154.                                        break;

  155.                               case 30: if(month==2 && year%4==0) { day=1; month=3; }

  156.                                        break;

  157.                               case 31: if(month==4 || month==6 || month==9 || month==11)

  158.                                        { day=1; month++; }

  159.                                           break;

  160.                               case 32: day=1;          month++;

  161.                                      if(month==13) { month=1; year++; }

  162.                              }

  163.                }

  164.            }

  165.        }

  166.      }

  167. }



  168. void Timer1ISR(void) interrupt 3 using 2            //定时器2中断,用于按键扫描

  169. {unsigned char keytmp;

  170. char tmp;



  171.   TH1=0x15; TL1=0xa0;      // 每30ms中断一次



  172.                  /***  当前显示的内容 ***/

  173.   if(show_status==0)   //当前正在设置闹钟项

  174.     { display[0]=cur_alarm_set;               display[1]=0xf;

  175.      display[2]=alarm[cur_alarm_set].h/10;      display[3]=alarm[cur_alarm_set].h%10;

  176.         display[4]=alarm[cur_alarm_set].m/10;      display[5]=alarm[cur_alarm_set].m%10;

  177.      display[6]=(alarm_wk>>cur_alarm_set)&1;  display[7]=(alarm_en>>cur_alarm_set)&1;

  178.      }



  179.   if(show_status==1 || show_status==3)         //当前显示或设置日期

  180.     { display[0]=year/10;       display[1]=year%10;    display[2]=month/10;  

  181.          display[3]=month%10;    display[4]=day/10;   display[5]=day%10;

  182.          display[6]=0xf;         display[7]=weekday;

  183.      }



  184.   if(show_status==2 || show_status==4)         //当前显示或设置时间

  185.     { display[0]=hour/10;      display[1]=hour%10;    display[2]=min/10;

  186.         display[3]=min%10;     display[4]=sec/10;       display[5]=sec%10;

  187.          display[6]=0xf;         display[7]=0xf;   //最后两后无显示

  188.         }



  189.                     /***  按键扫描及处理  ***/

  190.   keytmp=~(P1) & 0x0f;

  191.   if(keytmp==0)  { km=0;  kp=0; }  

  192.   else

  193.    { if(km==0)   km=1;  

  194.         else

  195.      { if(kp==0)

  196.         { kp=1;

  197.                 if(keytmp==1)                  //第一个按钮

  198.                  { if(sound) { alarm_stop=1; sound=0; }      //如果闹钟正响,按此键停止

  199.                    else if((show_status==1 || show_status==2) && led_on)  //正显示日期或时间

  200.                    { show_status=0;  cur_alarm_set=0;  attr=0x3f; }  //进入闹钟设置

  201.                       else if(show_status==0)                //如正在设置闹钟时间项

  202.                       { show_status=2;  new_alarm_info=1; attr=0xff; }  //返回当前时间显示

  203.                       return;

  204.                      }

  205.                   

  206.                 if(keytmp==2 && led_on)    //第二个按钮,仅当数码管打开时有效

  207.                  { switch(attr)

  208.                    { case 0xff: if(show_status==1) show_status=2;   //在显示时间与日期间切换

  209.                                 else if(show_status==2) show_status=1;

  210.                                       break;

  211.                         case 0x3f: if(show_status==0) cur_alarm_set=(cur_alarm_set+1)%8;

  212.                       else if(show_status==3)

  213.                              year=(year+1)%50;       //当前日期的“年”加1

  214.                      else if(show_status==4)

  215.                          hour=(hour+1)%24;       //当前时间的“时”加1

  216.                             break;   

  217.           case 0xcf: if(show_status==0)                     //闹钟设置的“时”加1

  218.   alarm[cur_alarm_set].h=(alarm[cur_alarm_set].h+1)%24;

  219.             else if(show_status==3)

  220.    { month++;                //当前日期的“月”加1

  221.      if(month==13) month=1; }

  222.                            else if(show_status==4)

  223.                             min=(min+1)%60;         //当前时间的“分”加1

  224.                                  break;

  225.                 case 0xf3: if(show_status==0)

  226.                            alarm[cur_alarm_set].m=(alarm[cur_alarm_set].m+1)%60;

  227.                          else if(show_status==3)

  228.                            {day++;                  //当前日期的“日”加1

  229.                             if(day==32) day=1; }

  230.                          else if(show_status==4)

  231.                            {count_down=8000;

  232.                             sec=(sec+1)%60; }        //当前时间的“秒”加1

  233.                               break;               

  234.             case 0xfd: if(show_status==0)

  235.                                       alarm_wk^=0x1<<cur_alarm_set; 周末标志位切换

  236.                                             break;

  237.                          case 0xfe: if(show_status==0)

  238.                                       alarm_en^=0x1<<cur_alarm_set; 启用标志位切换

  239.                        }       //end of switch(attr)

  240.                       return;

  241.                      }         //end of if(keytmp==1)



  242.                 if(keytmp==4)           //第三个按钮

  243.                  { switch(attr)

  244.                    { case 0xff: if(show_status==1 || show_status==2)

  245.                                    led_on=~led_on;        //打开或关闭数码管显示

  246.                                             break;

  247.                         case 0x3f: if(show_status==0)         //如果正在设置闹钟

  248.                                              { if(cur_alarm_set==0)  cur_alarm_set=7;

  249.                              else cur_alarm_set--;}

  250.                         else if(show_status==3)     //当前日期的“年”减1

  251.                                     { if(year==0)  year=49;  else year--; }

  252.                                  else if(show_status==4)     //当前时间的“时”减1

  253.                                              { tmp=hour-1;  if(tmp<0) hour=23;  else  hour=tmp;  }

  254.                                       break;

  255.                         case 0xcf: if(show_status==0)          //闹钟设置的“时”减1

  256.                                              { tmp=alarm[cur_alarm_set].h-1;

  257.                                                if(tmp<0)    alarm[cur_alarm_set].h=23;

  258.                                                   else         alarm[cur_alarm_set].h=tmp;

  259.                                                  }

  260.                           else if(show_status==3)         

  261.                                     { month--;             //当前日期的“月”减1

  262.                              if(month==0)  month=12; }

  263.                                             else if(show_status==4)         

  264.                            { tmp=min-1;           //当前时间的“分”减1

  265.                             if(tmp<0) min=59;   else  min=tmp;   }

  266.                                           break;

  267.                         case 0xf3: if(show_status==0)          //闹钟设置的“分钟”减1

  268.                                              { tmp=alarm[cur_alarm_set].m-1;

  269.                                                if(tmp<0)    alarm[cur_alarm_set].m=59;

  270.                                                   else         alarm[cur_alarm_set].m=tmp;

  271.                                                  }

  272.                           else if(show_status==3)     

  273.                                     { day--;                 //当前日期的“日”减1

  274.                              if(day==0)  day=31; }

  275.                                             else if(show_status==4)        

  276.                            { tmp=sec-1;             //当前时间的“秒”减1

  277.                              count_down=8000;

  278.                              if(tmp<0) sec=59;   else  sec=tmp; }

  279.                                             break;

  280.                         case 0xfd: if(show_status==0)                       //切换周末标志

  281.                                       alarm_wk^=0x1<<cur_alarm_set;

  282.                                             break;

  283.                         case 0xfe: if(show_status==0)                 //切换启用标志

  284.                                       alarm_en^=0x1<<cur_alarm_set;

  285.                        }       //end of switch(attr)

  286.                       return;

  287.                      }         //end of if(keytmp==2)



  288.                 if(keytmp==8 & led_on)    //第四个按钮,仅当数码管打开时有效

  289.                  { switch(attr)

  290.                       { case 0xff: if(show_status==1)               //如果当前显示日期

  291.                          show_status=3;       //切换到调准日期状态

  292.                                else if(show_status==2)    //如果当前显示时间

  293.                           show_status=4;      //切换到调准时间状态

  294.                                      attr=0x3f; break;               //第一、二个数码管闪烁

  295.                         case 0x3f: attr=0xcf; break;           //第三、四个数码管闪烁

  296.                         case 0xcf: attr=0xf3; break;           //第五、六个数码管闪烁

  297.                         case 0xf3: if(show_status==0) attr=0xfd;   //第七个数码管闪烁

  298.                         else if(show_status==3)

  299.                            { show_status=1; attr=0xff; }  //恢复正常显示日期

  300.                                  else if(show_status==4)

  301.                             { show_status=2; attr=0xff; }  //恢复正常显示时间

  302.                                           break;

  303.                         case 0xfd: if(show_status==0) attr=0xfe;     //第八个数码管闪烁

  304.                                    break;

  305.                         case 0xfe: if(show_status==0) attr=0x3f;     //第一、二个数码管闪烁

  306.                       }

  307.                      }   // end of if(keytmp==4)

  308.                }      // end of if(kp==0)

  309.         }         // end of if(km==0)

  310.    }            // end of if(keytmp!=0)

  311. }



  312. main()

  313. {unsigned char i;



  314.   hour=23;  min=58;  sec=30;  year=2;  month=4;  day=25;

  315.   count_down=8000;



  316.   flash=0;    attr=0xff;      led_on=1;

  317.   km=0;     kp=0;          show_status=2;     //加电后显示当前时间



  318.   new_alarm_info=0;  sound=0;   alarm_stop=0;  



  319.   for(i=0;i<8;i++)           //从EEPROM中读入8个闹钟设置

  320.     { alarm[i].h=myread(i*2);  alarm[i].m=myread(i*2+1); }

  321.       alarm_en=myread(i*2);        alarm_wk=myread(i*2+1);



  322.   IE=0;  IP=0;             //disable all interrupt and lower priority

  323.   TMOD=0x12;             //timer 0 is set to mode 2,auto_reloading,timer1,mode 1

  324.   TH0=6;    TL0=6;        //timer0 parameters for 0.125ms

  325.   TH1=0x15; TL1=0xa0;      //timer1 parameters for 30ms

  326.   TR0=1;    TR1=1;        //timer0 interrupt most important

  327.   ET0=1;    ET1=1;    EA=1;     //enable interrupt function



  328.   while(1)

  329.   { if(led_on)

  330.       for(i=0;i<8;i++)

  331.           { P2=0;

  332.          if(flash || attr&(0x80>>i))

  333.                  { P0=display[i];  P2=0x80>>i; DelayX1ms(1); }

  334.               }

  335.     else  P2=0;  



  336.     if(new_alarm_info)

  337.      { P2=0;                  //暂时关闭数码管

  338.           new_alarm_info=0;      //写入EEPROM

  339.        for(i=0;i<8;i++)  { mywrite(i*2,alarm[i].h);  mywrite(i*2+1,alarm[i].m); }

  340.           mywrite(i*2,alarm_en);   mywrite(i*2+1,alarm_wk);

  341.          }

  342.   }  

  343. }

  344.  
复制代码



</cur_alarm_set;
</cur_alarm_set;
</month;tmp++)
</count;i++)
回复

使用道具 举报

ID:105494 发表于 2016-2-19 15:15 | 显示全部楼层
嘿嘿嘿嘿呵呵嘿嘿
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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