找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机数码管万年历Proteus仿真+代码

[复制链接]
ID:321563 发表于 2019-5-20 18:42 | 显示全部楼层 |阅读模式
宝上的,给大家共享
TIM截图20190520183849.png
0.png

单片机源程序如下:
  1. #include <reg52.h>                 //调用单片机头文件
  2. #define uchar unsigned char  //无符号字符型 宏定义        变量范围0~255
  3. #define uint  unsigned int         //无符号整型 宏定义        变量范围0~65535

  4. #include "ds1302.h"



  5. //数码管段选定义      0     1    2    3    4    5          6         7          8           9        
  6. uchar code smg_du[]={0xa0,0x75,0x32,0x2a,0x32,0x25,0xb4,0xBa,0x20,0x28,
  7.                                            0x30,0x25,0xe4,0x23,0x64,0x74,0xff};         //断码


  8. //数码管位选定义
  9. uchar code smg_we[]={0xfe,0xbf,0xfb,0xef,0xf7,0xdf,0xfd,0x7f};//数码管位选定义
  10. uchar dis_smg[8] = {0xa0,0xbb,0x62,0x2a,0x39,0x2c,0x24,0xBa};        
  11. uchar smg_i = 8;    //显示数码管的个位数

  12. bit flag_200ms ;
  13. bit flag_100ms ;
  14. bit flag_500ms;     //500ms标志位

  15. sbit beep = P3^7;          //蜂鸣器定义
  16. bit flag_beep_en;
  17. uint clock_value;     //用作闹钟用的
  18. uchar flag_s = 0x03;//控制数码管闪烁的变量

  19. sbit dq   = P3^1;          //18b20 IO口的定义
  20. uint temperature ;    //温度变量
  21. uchar flag_nl;        //农历 阳历显示标志位

  22. uchar menu_1,menu_2,menu_i;


  23. sbit dat0 = B^0;
  24. sbit dat1 = B^1;
  25. sbit dat2 = B^2;
  26. sbit dat3 = B^3;
  27. sbit dat4 = B^4;
  28. sbit dat5 = B^5;
  29. sbit dat6 = B^6;
  30. sbit dat7 = B^7;


  31. //控制数码管闪烁的函数
  32. void smg_s(uchar dat)
  33. {
  34.         B = dat;
  35.         if(dat0 == 1)
  36.                 dis_smg[0] = 0xff;
  37.         if(dat1 == 1)
  38.                 dis_smg[1] = 0xff;
  39.         if(dat2 == 1)
  40.                 dis_smg[2] = 0xff;
  41.         if(dat3 == 1)
  42.                 dis_smg[3] = 0xff;
  43.         if(dat4 == 1)
  44.                 dis_smg[4] = 0xff;
  45.         if(dat5 == 1)
  46.                 dis_smg[5] = 0xff;
  47.         if(dat6 == 1)
  48.                 dis_smg[6] = 0xff;
  49.         if(dat7 == 1)
  50.                 dis_smg[7] = 0xff;
  51. }

  52. /***********************延时函数************************/
  53. void delay_uint(uint q)
  54. {
  55.         while(q--);
  56. }

  57. /***********************1ms延时函数*****************************/
  58. void delay_1ms(uint q)
  59. {
  60.         uint i,j;
  61.         for(i=0;i<q;i++)
  62.                 for(j=0;j<120;j++);
  63. }

  64. /********************独立按键程序*****************/
  65. uchar key_can;         //按键值

  66. void key()         //独立按键程序
  67. {
  68.         static uchar key_new;
  69.         key_can = 20;                   //按键值还原
  70.         P3 |= 0x78;                     //对应的按键IO口输出为1
  71.         if((P3 & 0x78) != 0x78)                //按键按下
  72.         {
  73.                 delay_1ms(1);                     //按键消抖动
  74.                 if(((P3 & 0x78) != 0x78) && (key_new == 1))
  75.                 {                                                //确认是按键按下
  76.                         key_new = 0;
  77.                         switch(P3 & 0x78)
  78.                         {
  79.                                 case 0xe0:  key_can = 1;  break;         //得到按键值
  80.                                 case 0x68:  key_can = 2;  break;         //得到按键值
  81.                                 case 0x58:  key_can = 3;  break;         //得到按键值
  82.                                 case 0x98:  key_can = 4;  break;         //得到按键值
  83.                         }
  84.                 }                        
  85.         }
  86.         else
  87.                 key_new = 1;        
  88. }

  89. #include "menu.h"


  90. /************菜单处理函数****************/
  91. void menu_dis()
  92. {
  93.         if((menu_1 > 0) && (menu_2 == 0))         //选择键面
  94.         {               
  95.                 if(flag_500ms == 1)
  96.                 {
  97.                         dis_smg[0] = smg_du[1];          //1
  98.                         dis_smg[1] = smg_du[10];  //A
  99.                         dis_smg[2] = smg_du[2];          //2
  100.                         dis_smg[3] = smg_du[10];  //A
  101.                         dis_smg[4] = smg_du[3];          //3
  102.                         dis_smg[5] = smg_du[10];  //A
  103.                         dis_smg[6] = smg_du[4];          //4
  104.                         dis_smg[7] = smg_du[10];  //A
  105.                         smg_i = 8;
  106.                 }
  107.                 else
  108.                         smg_s(flag_s);        
  109.         }
  110.         if((menu_1 == 1) && (menu_2 > 0))         //选设置时 分 秒
  111.         {
  112.                 smg_i = 8;
  113.                 if(flag_500ms == 1)
  114.                 {
  115.                         dis_smg[0] = smg_du[miao % 16];   //显示秒
  116.                         dis_smg[1] = smg_du[miao / 16];   //
  117.                         dis_smg[2] = 0x7f;
  118.                         dis_smg[3] = smg_du[fen % 16];   //显示分
  119.                         dis_smg[4] = smg_du[fen / 16];   //
  120.                         dis_smg[5] = 0x7f;
  121.                         dis_smg[6] = smg_du[shi % 16];   //显示秒
  122.                         dis_smg[7] = smg_du[shi / 16];   //
  123.                 }
  124.                 else
  125.                         smg_s(flag_s);               
  126.         }
  127.         if((menu_1 == 2) && (menu_2 > 0))         //选设置年 月 日
  128.         {
  129.                 smg_i = 8;
  130.                 if(flag_500ms == 1)
  131.                 {
  132.                         dis_smg[0] = smg_du[ri % 16];   //显示日
  133.                         dis_smg[1] = smg_du[ri / 16];   //
  134.                         dis_smg[2] = 0x7f;
  135.                         dis_smg[3] = smg_du[yue % 16];   //显示月
  136.                         dis_smg[4] = smg_du[yue / 16];   //
  137.                         dis_smg[5] = 0x7f;
  138.                         dis_smg[6] = smg_du[nian % 16];   //显示年
  139.                         dis_smg[7] = smg_du[nian / 16];   //
  140.                 }
  141.         }
  142.         if((menu_1 == 3) && (menu_2 > 0))         //设置星期 和 闹钟的开和关
  143.         {
  144.                 smg_i = 8;
  145.                 if(flag_500ms == 1)
  146.                 {
  147.                         dis_smg[0] = smg_du[week % 16];   //显示星期
  148.                         dis_smg[1] = smg_du[week / 16];   //
  149.                         dis_smg[2] = 0x7f;
  150.                         if(open1 == 1)           //开闹钟
  151.                         {
  152.                                 dis_smg[3] = 0xb0;           //OPEN
  153.                                 dis_smg[4] = 0x64;   
  154.                                 dis_smg[5] = 0x70;
  155.                                 dis_smg[6] = 0xa0;   
  156.                         }else
  157.                         {
  158.                                 dis_smg[3] = 0x74;           //OFF
  159.                                 dis_smg[4] = 0x74;   
  160.                                 dis_smg[5] = 0xa0;
  161.                                 dis_smg[6] = 0xFF;   
  162.                         }
  163.                         dis_smg[7] = 0xff;   
  164.                 }
  165.         }
  166.         if((menu_1 == 4) && (menu_2 > 0))         //选设置时 分 秒
  167.         {
  168.                 smg_i = 8;
  169.                 if(flag_500ms == 1)
  170.                 {
  171.                         dis_smg[0] = 0xff;   //
  172.                         dis_smg[1] = 0xff;   //
  173.                         dis_smg[2] = smg_du[fen1 % 16];   //显示分
  174.                         dis_smg[3] = smg_du[fen1 / 16];   //
  175.                         dis_smg[4] = 0x7f;
  176.                         dis_smg[5] = smg_du[shi1 % 16];   //显示秒
  177.                         dis_smg[6] = smg_du[shi1 / 16];   //
  178.                         dis_smg[7] = 0xff;
  179.                 }
  180.         }

  181. }

  182. /*************闹钟报警函数***************/
  183. void clock_dis()
  184. {
  185.         static uchar mm,value;
  186.         if(flag_100ms == 1)                  //100ms执行一次
  187.         {
  188.                 flag_100ms = 0;
  189.                 if(open1 == 1)        //如果闹钟打开
  190.                 {
  191.                         if((miao == 0) && (fen == fen1) && (shi == shi1))
  192.                         {               
  193.                                 flag_beep_en = 1;        //有报警 打开蜂鸣器响的标志位                                
  194.                         }               
  195.                         if(flag_beep_en == 1)        //闹钟以被打开
  196.                         {
  197.                                 clock_value++;
  198.                                 if(clock_value <= 30)         
  199.                                         beep = ~beep;           //蜂鸣器叫3秒
  200.                                 else if(clock_value > 30)
  201.                                 {
  202.                                         beep = 1;                    //蜂鸣器停1秒
  203.                                         if(clock_value > 40)
  204.                                         {
  205.                                                 clock_value = 0;

  206.                                         }
  207.                                 }
  208.                         }
  209.                 }        
  210.         }
  211. }

  212. /*************定时器0初始化程序***************/
  213. void time_init()         
  214. {
  215.         EA   = 1;                   //开总中断
  216.         TMOD = 0X01;          //定时器0、工作方式1
  217.         ET0  = 1;                  //开定时器0中断
  218.         TR0  = 1;                  //允许定时器0定时
  219. }

  220. /***********************18b20初始化函数*****************************/
  221. void init_18b20()
  222. {
  223.         bit q;
  224.         dq = 1;                                //把总线拿高
  225.         delay_uint(1);            //15us
  226.         dq = 0;                                //给复位脉冲
  227.         delay_uint(80);                //750us
  228.         dq = 1;                                //把总线拿高 等待
  229.         delay_uint(10);                //110us
  230.         q = dq;                                //读取18b20初始化信号
  231.         delay_uint(20);                //200us
  232.         dq = 1;                                //把总线拿高 释放总线
  233. }

  234. /*************写18b20内的数据***************/
  235. void write_18b20(uchar dat)
  236. {
  237.         uchar i;
  238.         for(i=0;i<8;i++)
  239.         {                                         //写数据是低位开始
  240.                 dq = 0;                         //把总线拿低写时间隙开始
  241.                 dq = dat & 0x01; //向18b20总线写数据了
  242.                 delay_uint(5);         // 60us
  243.                 dq = 1;                         //释放总线
  244.                 dat >>= 1;
  245.         }        
  246. }

  247. /*************读取18b20内的数据***************/
  248. uchar read_18b20()
  249. {
  250.         uchar i,value;
  251.         for(i=0;i<8;i++)
  252.         {
  253.                 dq = 0;                         //把总线拿低读时间隙开始
  254.                 value >>= 1;         //读数据是低位开始
  255.                 dq = 1;                         //释放总线
  256.                 if(dq == 1)                 //开始读写数据
  257.                         value |= 0x80;
  258.                 delay_uint(5);         //60us        读一个时间隙最少要保持60us的时间
  259.         }
  260.         return value;                 //返回数据
  261. }

  262. /*************读取温度的值 读出来的是小数***************/
  263. uint read_temp()
  264. {
  265.         uint value;
  266.         uchar low;                           //在读取温度的时候如果中断的太频繁了,就应该把中断给关了,否则会影响到18b20的时序
  267.         init_18b20();                   //初始化18b20
  268.         write_18b20(0xcc);           //跳过64位ROM
  269.         write_18b20(0x44);           //启动一次温度转换命令
  270.         delay_uint(50);                   //500us

  271.         init_18b20();                   //初始化18b20
  272.         
  273.         write_18b20(0xcc);           //跳过64位ROM
  274.         write_18b20(0xbe);           //发出读取暂存器命令
  275.         
  276.         EA = 0;
  277.         low = read_18b20();           //读温度低字节
  278.         value = read_18b20();  //读温度高字节
  279.         EA = 1;
  280.         value <<= 8;                   //把温度的高位左移8位
  281.         value |= low;                   //把读出的温度低位放到value的低八位中
  282.         value *= 0.625;               //转换到温度值 小数
  283.         return value;                   //返回读出的温度 带小数
  284. }

  285. /*************时钟显示***************/
  286. void init_1602_ds1302()
  287. {
  288.         static uchar value;
  289.         smg_i = 8;
  290.         value ++;
  291.         if(value <= 5*2)  //2秒
  292.         {
  293.                 dis_smg[0] = smg_du[miao % 16];   //显示秒
  294.                 dis_smg[1] = smg_du[miao / 16];   //
  295.                 dis_smg[2] = 0x3f;
  296.                 dis_smg[3] = smg_du[fen % 16];   //显示分
  297.                 dis_smg[4] = smg_du[fen / 16];   //
  298.                 dis_smg[5] = 0x3f;
  299.                 dis_smg[6] = smg_du[shi % 16];   //显示秒
  300.                 dis_smg[7] = smg_du[shi / 16];   //
  301.         }
  302.         else if(value <= 5*4)  //2秒
  303.         {
  304.                 dis_smg[0] = smg_du[ri % 16];   //显示日
  305.                 dis_smg[1] = smg_du[ri / 16];   //
  306.                 dis_smg[2] = 0x7f;
  307.                 dis_smg[3] = smg_du[yue % 16];   //显示月
  308.                 dis_smg[4] = smg_du[yue / 16];   //
  309.                 dis_smg[5] = 0x7f;
  310.                 dis_smg[6] = smg_du[nian % 16];   //显示年
  311.                 dis_smg[7] = smg_du[nian / 16];   //
  312.         }
  313.         else if(value <= 5*6)  //2秒
  314.         {
  315.                 dis_smg[0] = smg_du[week % 16];   //显示星期
  316.                 dis_smg[1] = smg_du[week / 16];   //
  317.                 dis_smg[2] = 0xff;               
  318.                 dis_smg[3] = smg_du[12];               
  319.                 dis_smg[4] = smg_du[temperature % 10] ;   //
  320.                 dis_smg[6] = smg_du[temperature / 100 % 10];   
  321.                 dis_smg[7] = 0xff;
  322.         }
  323.         else
  324.                 value = 0;
  325. }

  326. /*****************主函数********************/
  327. void main()
  328. {        
  329.         init_ds1302();                 //ds1302初始化
  330.         time_init();   //定时器初始化
  331.         init_menu();
  332.         while(1)
  333.         {
  334.                 key();                         //按键程序
  335.                 if(key_can < 10)
  336.                 {
  337.                         key_with();        
  338.                 }

  339.                 if(flag_200ms == 1)
  340.                 {        
  341.                         flag_200ms = 0;
  342.                         if(menu_1 == 0)
  343.                         {
  344.                                 read_time();                  //读时间
  345.                                 temperature = read_temp();        //先读出温度的值
  346.                                 init_1602_ds1302();   //显示时钟
  347.                         }
  348.                 }
  349.                 menu_dis();          //设置的对应的显示处理
  350.                 clock_dis();  //闹钟报警函数
  351.                 delay_1ms(1);
  352.         }
  353. }

  354. /*************定时器0中断服务程序***************/
  355. void time0_int() interrupt 1
  356. {        
  357.         static uint value;
  358.         static uchar ii;
  359.         TH0 = 0xF8;
  360.         TL0 = 0x30;     // 2ms         12M

  361.     value ++;
  362.         if((value % 50) == 0)         //100ms  
  363.                 flag_100ms = 1;         
  364.         if(value >= 250)
  365.         {
  366.                 value = 0;
  367.                 flag_500ms = ~flag_500ms;
  368.         }

  369.         P0 = 0xff;                         //消隐
  370.         P2 = smg_we[ii];                          //位选
  371.         P0 = dis_smg[ii];                 //段选            
  372.                
  373.         ii++;
  374.         if(ii >= smg_i)
  375.                 ii = 0;        
  376.         
  377. }
复制代码

所有资料51hei提供下载:
单片机万年历数码管显示 发送资料 SMG.7z (3.28 MB, 下载次数: 88)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:650913 发表于 2019-11-27 17:02 | 显示全部楼层
小白,想请问以下这个是AT89C51单片机吗?
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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