找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机电子时钟编程与仿真

[复制链接]
跳转到指定楼层
楼主
希望对各位有所帮助
仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)


单片机源程序如下:
  1. #include "AT89X51.h"

  2. #define LEDS 4
  3. //#include "reg51.h"
  4. //led灯选通信号
  5. unsigned char code Select[]={0x01,0x02,0x04,0x08,0x10,0x20};
  6. //unsigned char code LED_CODES[]= {0xc0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E};        //共阳
  7. unsigned char code LED_CODES[]= {0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07,0x7F,0x67};        //共阴

  8. #define        TH0_BASE                                (65536-10000)

  9. struct{
  10.         char CentiSecond;
  11.         char Second;
  12.         char Minute;
  13.         char Hour;
  14.         }Time;

  15. struct{
  16.         char CentiSecond;
  17.         char Second;
  18.         char Minute;
  19.         }Sec_Meter;

  20. unsigned char *Disp_fp,*Disp_fp_Set;

  21. unsigned char Mode = 0, Sec_Run =0, Time_Unit = 0;

  22. long delay_dat = 500;

  23. void delay(long time1)
  24. {

  25.   while(time1--);

  26.   return;


  27. }

  28. void Timer0_Int_Init(void)
  29. {
  30.         TMOD = 0x01;
  31.     TH0 = TH0_BASE / 256;
  32.         TL0 = TH0_BASE % 256;
  33.         ET0 = 1;
  34.         EA = 1;
  35.         TR0 = 1;
  36. }

  37. void Timer0_Int(void) interrupt TF0_VECTOR
  38. {
  39.     static char Bit_flag = 0;
  40.         char Bit_dat;

  41.         //----回复定时器初值--------------
  42.         TH0 = TH0_BASE / 256;
  43.         TL0 = TH0_BASE % 256;

  44.         //----时钟计时--------------------
  45.         Time.CentiSecond ++;
  46.         if(Time.CentiSecond > 99)
  47.         {
  48.                 Time.CentiSecond = 0;
  49.                 Time.Second ++;
  50.                 if(Time.Second > 59)
  51.                 {
  52.                         Time.Second = 0;
  53.                         Time.Minute ++;
  54.                         if(Time.Minute > 59)
  55.                         {
  56.                                 Time.Minute = 0;
  57.                                 Time.Hour ++;
  58.                                 if(Time.Hour > 23)        Time.Hour = 0;
  59.                         }
  60.                 }
  61.         }
  62.         if(Sec_Run == 1)
  63.         {
  64.                 Sec_Meter.CentiSecond ++;
  65.                 if(Sec_Meter.CentiSecond > 99)
  66.                 {
  67.                         Sec_Meter.CentiSecond = 0;
  68.                         Sec_Meter.Second ++;
  69.                         if(Sec_Meter.Second > 59)
  70.                         {
  71.                                 Sec_Meter.Second = 0;
  72.                                 Sec_Meter.Minute ++;
  73.                                 if(Sec_Meter.Minute > 59)
  74.                                 {
  75.                                         Sec_Meter.Minute = 0;
  76.                                 }
  77.                         }
  78.                 }       
  79.         }

  80.         //-----时间显示------------------
  81.         //注意:此处时间显示每次中断只显示1位,下次中断显示第二位,依次循环,由于中断周期为10ms,故4每位显示间隔为10ms,40ms一个循环
  82.         if(Bit_flag == 0)
  83.         {
  84.                 Bit_dat = *Disp_fp % 10;
  85.                 P3 = 0;
  86.                 P2=LED_CODES[Bit_dat];
  87.                 P3=Select[5-Bit_flag];
  88.                 Bit_flag = 1;
  89.                 //Disp_fp ++;
  90.         }
  91.         else if(Bit_flag == 1)
  92.         {
  93.                 Bit_dat = *Disp_fp / 10;
  94.                 P3 = 0;
  95.                 P2=LED_CODES[Bit_dat];
  96.                 P3=Select[5-Bit_flag];
  97.                 Bit_flag = 2;
  98.                 Disp_fp++;
  99.         }
  100.         else if(Bit_flag == 2)
  101.         {
  102.                 Bit_dat = *Disp_fp % 10;
  103.                 P3 = 0;
  104.                 if(Time.CentiSecond > 40)
  105.                         P2=LED_CODES[Bit_dat] | 0x80;
  106.                 else
  107.                         P2=LED_CODES[Bit_dat];
  108.                 P3=Select[5-Bit_flag];
  109.                 Bit_flag = 3;
  110.                 //Disp_fp++;
  111.         }
  112.         else if(Bit_flag == 3)
  113.         {
  114.                 Bit_dat = *Disp_fp / 10;
  115.                 P3 = 0;
  116.                 P2=LED_CODES[Bit_dat];
  117.                 P3=Select[5-Bit_flag];
  118.                 Bit_flag = 4;
  119.                 Disp_fp++;
  120.         }
  121.         else if(Bit_flag == 4)
  122.         {
  123.                 Bit_dat = *Disp_fp % 10;
  124.                 P3 = 0;
  125.                 if(Time.CentiSecond > 40)
  126.                         P2=LED_CODES[Bit_dat] | 0x80;
  127.                 else
  128.                         P2=LED_CODES[Bit_dat];
  129.                 P3=Select[5-Bit_flag];
  130.                 Bit_flag = 5;
  131.                 //Disp_fp++;
  132.         }
  133.         else if(Bit_flag == 5)
  134.         {
  135.                 Bit_dat = *Disp_fp / 10;
  136.                 P3 = 0;
  137.                 P2=LED_CODES[Bit_dat];
  138.                 P3=Select[5-Bit_flag];
  139.                 Bit_flag = 0;
  140.                 Disp_fp = Disp_fp_Set;
  141.         }
  142.         //---------------------------------------------
  143. }

  144. void Display()
  145. {;}

  146. //----------------------------------------------------------------------------
  147. void main()
  148. {
  149. char i=0;
  150. long int j;

  151. P1 = 0xFF;

  152. Timer0_Int_Init();

  153.         Disp_fp = (unsigned char *)&Time.Second;
  154.         Disp_fp_Set = Disp_fp;

  155. while(1)
  156. {

  157.         if(!P1_0)
  158.         {
  159.                 while(!P1_0);
  160.                 Mode += 1;if(Mode > 2) Mode = 0;

  161.                 if(Mode == 0)                //正常时钟模式
  162.                 {
  163.                         Disp_fp = (unsigned char *)&Time.Second;
  164.                         Disp_fp_Set = Disp_fp;
  165.                 }
  166.                 else if(Mode == 1)        //秒表模式
  167.                 {
  168.                         Disp_fp = (unsigned char *)&Sec_Meter.CentiSecond;
  169.                         Disp_fp_Set = Disp_fp;
  170.                 }
  171.                 else if(Mode == 2)        //校时模式
  172.                 {
  173.                         Disp_fp = (unsigned char *)&Time.Second;
  174.                         Disp_fp_Set = Disp_fp;
  175.                 }
  176.         }
  177.         if(Mode == 1)        //秒表模式       
  178.         {
  179.                 if(!P1_1)
  180.                 {
  181.                         while(!P1_1);
  182.                         Sec_Run ++; if(Sec_Run > 2)Sec_Run = 0;
  183.                 }
  184.                
  185.                 if(Sec_Run == 0)                //清表
  186.                 {
  187.                         Sec_Meter.CentiSecond = 0;
  188.                         Sec_Meter.Second = 0;
  189.                         Sec_Meter.Minute = 0;
  190.                 }
  191.                 else if(Sec_Run == 1);        //计时
  192.                 else if(Sec_Run == 2);        //停止
  193.         }
  194.         else if(Mode == 2)                //校时模式
  195.         {
  196.                 if(!P1_1)
  197.                 {
  198.                         while(!P1_1);
  199.                         Time_Unit ++; if(Time_Unit > 2)Time_Unit = 0;
  200.                 }
  201.                 if(!P1_2)
  202.                 {
  203.                         while(!P1_2);
  204.                         if(Time_Unit == 0)        //校时
  205.                         {
  206.                                 Time.Hour ++;if(Time.Hour > 23) Time.Hour = 0;
  207.                         }
  208.                         else if(Time_Unit == 1)        //校分
  209.                         {
  210.                                 Time.Minute ++;if(Time.Minute > 59) Time.Minute = 0;
  211.                         }
  212.                         else if(Time_Unit == 2)        //校秒
  213.                         {
  214.                                 Time.Second ++;if(Time.Second > 59) Time.Second = 0;
  215.                         }
  216.                 }
  217.                 if(!P1_3)
  218.                 {
  219.                         while(!P1_3);
  220.                         if(Time_Unit == 0)        //校时
  221.                         {
  222.                                 Time.Hour --;if(Time.Hour < 0) Time.Hour = 23;
  223.                         }
  224.                         else if(Time_Unit == 1)        //校分
  225.                         {
  226.                                 Time.Minute --;if(Time.Minute < 0) Time.Minute = 59;
  227.                         }
  228.                         else if(Time_Unit == 2)        //校秒
  229.                         {
  230.                                 Time.Second --;if(Time.Second < 0) Time.Second = 59;
  231.                         }
  232.                 }       
  233.        
  234.         }

  235. //P2=0;
  236. /*
  237.   i = Hour / 10;
  238.   P2=LED_CODES[i];
  239.   P3=Select[0];

  240.   i = Hour % 10;
  241.   P2=LED_CODES[i];
  242.   P3=Select[1];

  243.   P3 = 0;
  244. ……………………

  245. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:

Proteus_51_Demo.rar (53.53 KB, 下载次数: 25)


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

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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