找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 5066|回复: 2
收起左侧

智能秒表(单片机内部EEPROM)的设计

[复制链接]
ID:189697 发表于 2017-7-4 20:19 | 显示全部楼层 |阅读模式
这是我做的一个基于51单片机的秒表程序(数码管显示),具有以下功能:智能秒表(使用内部EEPROM)
(1)启动/停止功能
(2)复位/间隔功能  
(3)两种工作模式  
(4)查看数据功能      
(5)断电记忆功能  

智能秒表2.jpg

1、模式1:累积计时模式
1)按按键1可实现秒表开始、停止、再开始、再停止计时(时间累计)
(2)计时完毕,按按键2计时归零。

2、模式2:间隔计时模式
(1) 按按键1开始计时;
(2) 按按键2一次暂停秒表并显示当前时间。间隔时间显示2秒后秒表将持续计时。(例如在3秒钟时秒表被暂停,2秒钟后秒表从5秒开始继续计时。显示2秒是为了方便记录当前时间);
(3) 在下一点终止时再按按键2记录此时相关信息,如此重复;
(4) 当计时停止时按按键1;
(5)可通过按键3查看前面操作所记录的时间,最多可查看10个记录;
(6) 计时完毕,按按键2键计时归零。
智能秒表1.jpg

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载):
0.png

0.png

七、调试方法和步骤
1、通电前先用万用表欧姆档(1KΩ档)测量电源两端正反向电阻,正向电阻约80KΩ,反向电阻约6.5KΩ。如果电阻很小,则说明有短路性故障,不能通电,排除短路故障后才能通电,如果电阻为无穷大,则有开路性故障。
2、通电后,测量单片机40脚与20脚之间的电压应该有5V左右,如果电压不正常,则说明单片机供电不正常,应重点检查单片机供电电路。
3、测量单片机晶振引脚18和19脚电压,应该在1.9~2.3V左右,两个引脚电压相差0.2V左右,如果这两个引脚电压不正常,说明晶振电路没有起振,重点检查晶振和补偿电容是否焊接好。
4、测量单片机复位引脚9脚,正常工作时电压为0V,按复位键时为5V,松开手后由5V逐渐降至0V。如果该脚电压不正常,说明复位电路有故障,需要检查该脚的电阻、电容的连接和焊接是否正确。
5、以上步骤的测量如果正常,则多数是单片机的软件问题,可以通过编写几个小程序单独调试I/O口、按键、数码管和存储器的功能,测试到底是软件问题还是硬件问题。


全部资料带完整论文下载(word格式 可编辑):
项目1智能秒表2(内部EEPROM).rar (1.35 MB, 下载次数: 128)

评分

参与人数 2黑币 +53 收起 理由
18677304384 + 3 很给力!
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:1 发表于 2017-7-4 23:49 | 显示全部楼层
楼主的源程序如下:
  1. /********************************/
  2. /*   智能秒表(使用内部EEPROM) */
  3. /*------------------------------*/
  4. /**(1)启动/停止功能          **/
  5. /**(2)复位/间隔功能          **/
  6. /**(3)两种工作模式           **/
  7. /**(4)查看数据功能           **/
  8. /**(5)断电记忆功能           **/
  9. /********************************/
  10. #include <reg52.h>
  11. #include <intrins.h>
  12. #define uint unsigned int
  13. #define uchar unsigned char

  14. #define RdCommand 0x01                  //定义内部EEPROM存储器部分
  15. #define PrgCommand 0x02
  16. #define EraseCommand 0x03
  17. #define Error 1
  18. #define Ok 0
  19. #define WaitTime 0x01
  20. sfr ISP_DATA=0xe2;  
  21. sfr ISP_ADDRH=0xe3;
  22. sfr ISP_ADDRL=0xe4;
  23. sfr ISP_CMD=0xe5;
  24. sfr ISP_TRIG=0xe6;
  25. sfr ISP_CONTR=0xe7;

  26. sbit key1=P2^0;              //定义"启动/暂停"按钮
  27. sbit key2=P2^1;              //定义"复位"按钮
  28. sbit key3=P2^2;                                 //定义"模式"按钮
  29. sbit key4=P2^3;                                 //定义"查看"按钮
  30. sbit key5=P1^5;                                 //定义"清除"按钮

  31. sbit wei1=P2^4;                                 //定义位选
  32. sbit wei2=P2^5;
  33. sbit wei3=P2^6;               
  34. sbit wei4=P2^7;

  35. bit flag_start,flag_mode,flag_2sec,flag_CLR;
  36. uchar temp,temp1,aa,aa1,shi,ge,shifen,baifen,num,i=0;
  37. uchar time_data1[11],time_data2[11];
  38. uchar code table[]={                  //共阴极数码管编码
  39. 0x3f,0x06,0x5b,0x4f,
  40. 0x66,0x6d,0x7d,0x07,
  41. 0x7f,0x6f,0x77,0x7c,
  42. 0x39,0x5e,0x79,0x71,0x40};        

  43. void display(uchar shi,uchar ge,uchar shifen,uchar baifen);  //声明显示子函数
  44. void delay(uint z);           //声明延时子函数
  45. void init();                  //声明初始化函数
  46. void key();                                          //声明键扫描函数
  47. void display_process();                  //声明显示处理函数
  48. void delayus();                                  //声明延时微秒函数                          
  49. void data_read();                          //声明内部EEPROM数据读出函数
  50. void clear();                                  //声明清除EEPROM数据函数

  51. void ISP_IAP_enable(void);          //声明EEPROM操作函数
  52. void ISP_IAP_disable(void);
  53. void ISPgoon(void);
  54. uchar byte_read(uint byte_addr);
  55. void SectorErase(uint sector_addr);
  56. void byte_write(uint byte_addr, uchar original_data);
  57. void memory_read();                          //声明EEPROM读函数
  58. void memory_write();                  //声明EEPROM写函数

  59. void main()                                          //主函数
  60. {       
  61.         init();                          //调用初始化子程序
  62.         while(1)
  63.         {                      
  64.           key();                                                   //调用按键扫描函数
  65.           if(num==0&&i!=0)                                   //如果一开机就按查看键
  66.             memory_read();                                   //调用EEPROM数据读出函数
  67.           if(flag_CLR==1)                                   //如果按了清除键
  68.                 clear();                                           //调用清除EEPROM数据函数
  69.           display_process();                           //调用显示处理函数
  70.           display(shi,ge,shifen,baifen);   //调用显示函数   
  71.         }
  72. }

  73. void delay(uint z)                     //延时毫秒函数
  74. {
  75.         uint x,y;
  76.         for(x=z;x>0;x--)
  77.                 for(y=110;y>0;y--);
  78. }

  79. void display(uchar shi,uchar ge,uchar shifen,uchar baifen)   //显示程序
  80. {       
  81.                 P0=table[shi];        //使用动态扫描的方法实现数码管显示
  82.                 wei1=0;
  83.                 delay(1);
  84.                 wei1=1;

  85.                 if(flag_CLR==0)                  //正常显示时,第二位显示小数点
  86.                   P0=table[ge]|0x80;
  87.                 else                                  //按清除键时,不显示小数点
  88.                   P0=table[ge];
  89.                 wei2=0;
  90.                 delay(1);            
  91.                 wei2=1;
  92.    
  93.         P0=table[shifen];      //使用动态扫描的方法实现数码管显示
  94.                 wei3=0;
  95.                 delay(1);
  96.                 wei3=1;

  97.                 P0=table[baifen];
  98.                 wei4=0;
  99.                 delay(1);            
  100.                 wei4=1;       
  101. }

  102. void init()                     //初始化子程序
  103. {       
  104.         temp=0;
  105.         TMOD=0x01;                  //使用定时器T0的方式1
  106.         TH0=(65536-10000)/256;
  107.         TL0=(65536-10000)%256;      //定时10ms中断一次
  108.         EA=1;                       //中断总允许
  109.         ET0=1;                      //允许定时器T0中断       
  110. }

  111. void key()                                                //按键扫描函数
  112. {
  113.         uchar j;
  114.         if(key1==0)                 //检测"启动"按钮是否按下
  115.          {
  116.                 delay(5);               //延时去抖动
  117.                 if(key1==0)             //再次检测"启动"按钮是否按下
  118.                  {       
  119.            while(!key1)         //松手检测(按下时数码管不灭)
  120.                     {
  121.                           display_process();                          //调用显示处理函数
  122.                   display(shi,ge,shifen,baifen);  //调用显示函数
  123.                         }
  124.                    flag_start=~flag_start;
  125.                    if(flag_start==1)
  126.                          TR0=1;                    //启动定时器开始工作
  127.                    else
  128.                          TR0=0;
  129.                  }
  130.          }
  131.                 if(key2==0)                    //检测"间隔"按钮是否按下   
  132.                 {
  133.                         delay(5);         
  134.                         if(key2==0)      
  135.                         {       
  136.                                 while(!key2)                                        //松手检测
  137.                                 {
  138.                                  display_process();                            //调用显示处理函数
  139.                      display(shi,ge,shifen,baifen); //调用显示函数
  140.                                 }
  141.                                 if(flag_mode==0||i!=0)
  142.                                  {  
  143.                                   shi=0;              
  144.                                   ge=0;
  145.                                   shifen=0;
  146.                                   baifen=0;
  147.                                   aa=0;
  148.                                   temp=0;
  149.                                   TR0=0;
  150.                                   i=0;
  151.                                   num=0;
  152.                                   for(j=0;j<=10;j++)
  153.                                   {
  154.                                     time_data1[j]=0;
  155.                                     time_data2[j]=0;
  156.                                   }
  157.                                  }
  158.                             else
  159.                              {
  160.                                    num++;
  161.                                    if(num==11)
  162.                                      num=1;
  163.                                    memory_write();
  164.                                  }
  165.                         }
  166.                 }
  167.                 if(key3==0)
  168.                 {
  169.                   delay(5);
  170.                   if(key3==0)
  171.                    {
  172.                      while(!key3)                                            //松手检测
  173.                            {
  174.                                 display_process();                            //调用显示处理函数
  175.                     display(shi,ge,shifen,baifen);  //调用显示函数
  176.                           }
  177.                          flag_mode=~flag_mode;
  178.                    }
  179.                 }
  180.            if(key4==0)
  181.                 {
  182.                   delay(5);
  183.                   if(key4==0)
  184.                    {
  185.                      while(!key4)                                            //松手检测
  186.                           {
  187.                             display_process();                            //调用显示处理函数
  188.                     display(shi,ge,shifen,baifen);  //调用显示函数
  189.                           }
  190.                          TR0=0;
  191.                          i++;
  192.                          if(i==11)
  193.                           i=1;
  194.                          data_read();
  195.                    }
  196.                 }
  197.            if(key5==0)
  198.                 {
  199.                   delay(5);
  200.                   if(key5==0)
  201.                    {
  202.                      while(!key5)                                            //松手检测
  203.                           {
  204.                             display_process();                            //调用显示处理函数
  205.                     display(shi,ge,shifen,baifen);  //调用显示函数
  206.                           }
  207.                          flag_CLR=~flag_CLR;
  208.                    }
  209.                 }
  210. }               

  211. void display_process()                            //显示处理函数
  212. {
  213.   shifen=aa%100/10;                                                    //正常走秒的处理
  214.   baifen=aa%10;
  215.   shi=temp%100/10;
  216.   ge=temp%10;
  217.   if(flag_mode==1&&num!=0&&flag_2sec==0)        //间隔2秒停顿的处理
  218.    {
  219.          shi=time_data2[num]/10;
  220.      ge=time_data2[num]%10;
  221.          shifen=time_data1[num]/10;
  222.          baifen=time_data1[num]%10;
  223.          if(temp-time_data2[num]==2)
  224.            flag_2sec=~flag_2sec;
  225.    }
  226.   if(i!=0)                                                                    //查看存储数据的处理
  227.         {
  228.           shi=time_data2[i]/10;
  229.           ge=time_data2[i]%10;
  230.           shifen=time_data1[i]/10;
  231.           baifen=time_data1[i]%10;
  232.     }
  233. }

  234. void timer0() interrupt 1         //定时器T0中断服务函数
  235. {
  236.         TH0=(65536-10000)/256;   //重新赋初值
  237.         TL0=(65536-10000)%256;
  238.         aa++;                    //中断一次10ms变量aa的值加1
  239.         if(aa==100)              //中断100次后,定时时间100*10ms=1000ms=1s,变量temp的值加1
  240.         {
  241.           aa=0;
  242.           temp++;
  243.           if(temp==100)          //秒表到达100s后回零
  244.                 temp=0;
  245.         }       
  246. }

  247. /* EEPROM存储器操作函数 */
  248. /* ================================= */
  249. void ISP_IAP_enable(void)
  250. {
  251. EA = 0;      
  252. ISP_CONTR = ISP_CONTR & 0x18;      
  253. ISP_CONTR = ISP_CONTR | WaitTime;
  254. ISP_CONTR = ISP_CONTR | 0x80;      
  255. }
  256. /* ================================= */
  257. void ISP_IAP_disable(void)
  258. {
  259. ISP_CONTR=ISP_CONTR&0x7f;
  260. ISP_TRIG=0x00;
  261. EA=1;   
  262. }
  263. /* ==================================== */
  264. void ISPgoon(void)
  265. {
  266. ISP_IAP_enable();   
  267. ISP_TRIG = 0x46;  
  268. ISP_TRIG = 0xb9;  
  269. _nop_();
  270. }
  271. /* ============================================ */
  272. uchar byte_read(uint byte_addr)
  273. {
  274. ISP_ADDRH = (uchar)(byte_addr >> 8);
  275. ISP_ADDRL = (uchar)(byte_addr & 0x00ff);
  276. ISP_CMD   = ISP_CMD & 0xf8;   
  277. ISP_CMD   = ISP_CMD | RdCommand;
  278. ISPgoon();      
  279. ISP_IAP_disable();   
  280. return (ISP_DATA);   
  281. }
  282. /* ========================================== */
  283. void SectorErase(uint sector_addr)
  284. {
  285. uint iSectorAddr;
  286. iSectorAddr = (sector_addr & 0xfe00);
  287. ISP_ADDRH   = (uchar)(iSectorAddr >> 8);
  288. ISP_ADDRL   = 0x00;
  289. ISP_CMD     = ISP_CMD & 0xf8;   
  290. ISP_CMD     = ISP_CMD | EraseCommand;
  291. ISPgoon();      
  292. ISP_IAP_disable();   
  293. }
  294. /* ============================================ */
  295. void byte_write(uint byte_addr, uchar original_data)
  296. {
  297. ISP_ADDRH = (uchar)(byte_addr >> 8);  
  298. ISP_ADDRL = (uchar)(byte_addr & 0x00ff);
  299. ISP_CMD   = ISP_CMD & 0xf8;   
  300. ISP_CMD   = ISP_CMD | PrgCommand;
  301. ISP_DATA  = original_data;  
  302. ISPgoon();      
  303. ISP_IAP_disable();   
  304. }

  305. void data_read()                                         //读数组数据函数
  306. {
  307.   aa=time_data1[i];
  308.   temp=time_data2[i];
  309. }

  310. void memory_write()                                          //写EEPROM数据函数
  311. {
  312.   aa1=aa;
  313.   temp1=temp;
  314.   time_data1[num]=aa1;
  315.   time_data2[num]=temp1;
  316.   if(num==1)
  317.     SectorErase(0x2000);              //擦除扇区
  318.   byte_write(0x2000+num,aa1);         //重新写入数据
  319.   byte_write(0x2000+num+10,temp1);    //重新写入数据   
  320.   flag_2sec=0;
  321. }

  322. void memory_read()                                                  //读出EEPROM数据函数
  323. {
  324.   time_data1[i]=byte_read(0x2000+i);
  325.   time_data2[i]=byte_read(0x2000+i+10);
  326. }

  327. void clear()                                                  //清除EEPROM数据函数
  328. {
  329.   uchar j,k;
  330.   for(k=0;k<21;k++)
  331.     {
  332.           byte_write(0x2000+k,0x00);
  333.           display(16,16,16,16);
  334.     }
  335.   for(j=0;j<=10;j++)
  336.         {
  337.           time_data1[j]=0;
  338.           time_data2[j]=0;
  339.           display(16,16,16,16);
  340.         }
  341. }
复制代码
回复

使用道具 举报

ID:89286 发表于 2017-7-5 04:55 | 显示全部楼层
thanks for sharing
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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