找回密码
 立即注册

QQ登录

只需一步,快速开始

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

51单片机数码管显示温度并报警,只允许用三个按键和两个数码管(仿真+程序)

[复制链接]
跳转到指定楼层
楼主
      楼主最近做到一个作业,要求用51单片机做一个风扇温控检测系统,其功能大概就是:当温度达到设定值时候,风扇电机开始转动,以达到散热的目的,题目本身没有什么意思,有意思的是想要用尽可能少的元器件来完成这项工作,要求使用两个数码管,其他没说,所以我就加了3个按键,大体思路是这样的,开机以后,正常显示温度,按下SET键位时候,进温度设置,两个按键+和-设置数值。再按一次SET保存并退出。
     最有意思的在于用了两个标志位,这两个标志位简直没把我绕晕。现在完成了,发出来大家一起看看,不在于代码多好,在于一种解决问题的方法。我把文件发上来了,大家一起学习交流。

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


单片机源程序如下:
  1. #include "reg52.h"
  2. #include <intrins.h>

  3. #define  _Nop()  _nop_()        //定义空指令

  4. /*-----------------------------------------------
  5.                   IO口定义
  6. -----------------------------------------------*/
  7. sbit LED_R = P1^6;     //红灯
  8. sbit BEEP = P2^1;      //蜂鸣器
  9. sbit MOTOR = P2^2;     //电机

  10. sbit KEY_ADD = P2^5;    //加
  11. sbit KEY_DEC = P2^4;    //减
  12. sbit KEY_SET = P2^3;    //确认

  13. sbit ONE = P2^6;       //个位位选
  14. sbit TEN = P2^7;       //十位位选

  15. sbit DQ=P1^7;            //DS18B20数据

  16. sbit SDA=P1^1;            //模拟I2C数据传送位
  17. sbit SCL=P1^0;            //模拟I2C时钟控制位

  18.                           

  19.    

  20. /*-----------------------------------------------
  21.                     定义变量
  22. -----------------------------------------------*/
  23. unsigned char Set_Flag=0;        //
  24. unsigned char Set_Value=40;         //设定值
  25. unsigned char Tempature = 0;     //温度
  26. unsigned char num=40;            //数字
  27. unsigned char table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f}; //数码管段选
  28. unsigned char Data=0;
  29. char Temp[12];
  30. bit ack;                      //应答标志位
  31. /*-----------------------------------------------
  32.                     延时函数
  33. -----------------------------------------------*/
  34. void Delay_us(unsigned char t)
  35. {
  36.         while(t--);

  37. }

  38. void Delay_ms(unsigned char t)
  39. {
  40.         int i,j;
  41.         for(i=t;i>0;i--)
  42.                 for(j=125;j>0;j--);
  43. }

  44. void delay_18B20(unsigned int i)//微秒级延时
  45. {
  46. while(i--);
  47. }

  48. /*-----------------------------------------------
  49.                   DS18B20初始化
  50. -----------------------------------------------*/
  51. void ds1820rst(void)                /*ds1820复位*/
  52. {
  53.         unsigned char x=0;
  54.         DQ = 1;                         //DQ复位
  55.         delay_18B20(5);         //延时
  56.         DQ = 0;                         //DQ拉低
  57.         delay_18B20(100);         //精确延时大于480us
  58.         DQ = 1;                         //拉高
  59.         delay_18B20(40);
  60. }

  61. unsigned char ds1820rd(void)//从18b20读取数据
  62. {
  63.         unsigned char i=0;
  64.         unsigned char dat = 0;
  65.         for (i=8;i>0;i--)
  66.         {
  67.                 DQ = 0;                         //给脉冲信号
  68.                  dat>>=1;
  69.                  DQ = 1;                         //给脉冲信号
  70.                  if(DQ) dat|=0x80;
  71.                  delay_18B20(10);
  72.           }
  73.            return(dat);
  74. }
  75. void ds1820wr(unsigned char wdata)/*写数据*/
  76. {        
  77.         unsigned char i=0;
  78.         for (i=8; i>0; i--)
  79.         {
  80.                 DQ = 0;
  81.                  DQ = wdata&0x01;
  82.                  delay_18B20(25);
  83.                  DQ = 1;
  84.                  wdata>>=1;
  85.         }
  86. }
  87. //读取DS18B20温度值
  88. //入口参数:无
  89. //返回值:温度值
  90. unsigned int read_18b20(void)/*读取温度值并转换*/
  91. {   
  92.         unsigned char a,b;
  93.         unsigned char tem;
  94.         ds1820rst();               
  95.         ds1820wr(0xcc);//*跳过读序列号*/
  96.         ds1820wr(0x44);//*启动温度转换*/
  97.         ds1820rst();
  98.         ds1820wr(0xcc);//*跳过读序列号*/
  99.         ds1820wr(0xbe);//*读取温度*/
  100.         a=ds1820rd();
  101.         b=ds1820rd();
  102.         tem=((b<<4)|(a>>4));        //将高低字节组合成字
  103.   //Delay_ms(50);
  104.         return tem;        //完成温度转换
  105. }


  106. /*-----------------------------------------------
  107.                 数码管
  108. -----------------------------------------------*/
  109. //共阴数码管
  110. //动态扫描
  111. void display(unsigned char Data)
  112. {
  113.                   //P3 = 0;
  114.                         ONE = 1; //打开个位位选
  115.       TEN = 0;
  116.                         P3 = table[Data/10];//送个位
  117.                         Delay_ms(5);//延时
  118.       P3 = 0;        //防止重影
  119.       //Delay_us(5);
  120.   
  121.                         TEN = 1;
  122.       ONE = 0;
  123.                         P3 = table[Data%10];
  124.                         Delay_ms(5);
  125.       P3 = 0;
  126.       //Delay_us(5);
  127. }

  128. /*-----------------------------------------------
  129.                    按键处理函数
  130. -----------------------------------------------*/
  131. char S_F=0;
  132. unsigned char Key_Value(void)
  133. {
  134.   if(!KEY_ADD || !KEY_DEC||!KEY_SET)  //   检测是否有任意按键按下
  135.   {     
  136.     Delay_ms(20);
  137.     Set_Flag = 1;                       //切换到显示设置值   
  138.     if(!KEY_ADD)  //如果检测到低电平,说明按键按下
  139.       {
  140.         Delay_ms(10); //延时去抖,一般10-20ms
  141.         if(!KEY_ADD)     //再次确认按键是否按下,没有按下则退出
  142.         {
  143.         while(!KEY_ADD);//如果确认按下按键等待按键释放,没有释放则一直等待
  144.                {
  145.                       if(num<99)    //加操作
  146.                   num++;
  147.                             }
  148.         }
  149.             }

  150.     if(!KEY_DEC)  //如果检测到低电平,说明按键按下
  151.         {
  152.           Delay_ms(10); //延时去抖,一般10-20ms
  153.           if(!KEY_DEC)     //再次确认按键是否按下,没有按下则退出
  154.           {
  155.             while(!KEY_DEC);//如果确认按下按键等待按键释放,没有释放则一直等待
  156.           {
  157.             if(num>0)  //减操作
  158.                   num--;
  159.           }
  160.         }
  161.       }

  162.   if(!KEY_SET)  //如果检测到低电平,说明按键按下
  163.       {
  164.         Delay_ms(10); //延时去抖,一般10-20ms
  165.         if(!KEY_SET)     //再次确认按键是否按下,没有按下则退出
  166.         {
  167.                   //Set_Flag = 0;
  168.           while(!KEY_SET);//如果确认按下按键等待按键释放,没有释放则一直等待
  169.                {
  170.              //将当前值作为设定值
  171.             
  172.             //Set_Flag = 0;    //切换到显示温度值
  173.                             }
  174.          if(S_F)
  175.                                          Set_Flag=0;
  176.         }

  177.       }
  178.     }
  179.                
  180.     Set_Value = num;
  181.     return (num);

  182. }

  183. /*-----------------------------------------------
  184.                       24C02初始化
  185. -----------------------------------------------*/
  186. /*------------------------------------------------
  187.                     启动总线
  188. ------------------------------------------------*/
  189. void Start_I2c()
  190. {
  191.   SDA=1;   //发送起始条件的数据信号
  192.   _Nop();
  193.   SCL=1;
  194.   _Nop();    //起始条件建立时间大于4.7us,延时
  195.   _Nop();
  196.   _Nop();
  197.   _Nop();
  198.   _Nop();   
  199.   SDA=0;     //发送起始信号
  200.   _Nop();    //起始条件锁定时间大于4μ
  201.   _Nop();
  202.   _Nop();
  203.   _Nop();
  204.   _Nop();      
  205.   SCL=0;    //钳住I2C总线,准备发送或接收数据
  206.   _Nop();
  207.   _Nop();
  208. }
  209. /*------------------------------------------------
  210.                     结束总线
  211. ------------------------------------------------*/
  212. void Stop_I2c()
  213. {
  214.   SDA=0;    //发送结束条件的数据信号
  215.   _Nop();   //发送结束条件的时钟信号
  216.   SCL=1;    //结束条件建立时间大于4μ
  217.   _Nop();
  218.   _Nop();
  219.   _Nop();
  220.   _Nop();
  221.   _Nop();
  222.   SDA=1;    //发送I2C总线结束信号
  223.   _Nop();
  224.   _Nop();
  225.   _Nop();
  226.   _Nop();
  227. }




  228. /*----------------------------------------------------------------
  229.                  字节数据传送函数               
  230. 函数原型: void  SendByte(unsigned char c);
  231. 功能:  将数据c发送出去,可以是地址,也可以是数据,发完后等待应答,并对
  232.      此状态位进行操作.(不应答或非应答都使ack=0 假)     
  233.      发送数据正常,ack=1; ack=0表示被控器无应答或损坏。
  234. ------------------------------------------------------------------*/
  235. void  SendByte(unsigned char c)
  236. {
  237. unsigned char BitCnt;

  238. for(BitCnt=0;BitCnt<8;BitCnt++)  //要传送的数据长度为8位
  239.     {
  240.      if((c<<BitCnt)&0x80)SDA=1;   //判断发送位
  241.        else  SDA=0;               
  242.      _Nop();
  243.      SCL=1;               //置时钟线为高,通知被控器开始接收数据位
  244.       _Nop();
  245.       _Nop();             //保证时钟高电平周期大于4μ
  246.       _Nop();
  247.       _Nop();
  248.       _Nop();         
  249.      SCL=0;
  250.     }
  251.    
  252.     _Nop();
  253.     _Nop();
  254.     SDA=1;               //8位发送完后释放数据线,准备接收应答位
  255.     _Nop();
  256.     _Nop();   
  257.     SCL=1;
  258.     _Nop();
  259.     _Nop();
  260.     _Nop();
  261.     if(SDA==1)ack=0;     
  262.        else ack=1;        //判断是否接收到应答信号
  263.     SCL=0;
  264.     _Nop();
  265.     _Nop();
  266. }







  267. /*----------------------------------------------------------------
  268.                  字节数据传送函数               
  269. 函数原型: unsigned char  RcvByte();
  270. 功能:  用来接收从器件传来的数据,并判断总线错误(不发应答信号),
  271.      发完后请用应答函数。  
  272. ------------------------------------------------------------------*/        
  273. unsigned char  RcvByte()
  274. {
  275.   unsigned char retc;
  276.   unsigned char BitCnt;
  277.   
  278.   retc=0;
  279.   SDA=1;             //置数据线为输入方式
  280.   for(BitCnt=0;BitCnt<8;BitCnt++)
  281.       {
  282.         _Nop();           
  283.         SCL=0;       //置时钟线为低,准备接收数据位
  284.         _Nop();
  285.         _Nop();      //时钟低电平周期大于4.7us
  286.         _Nop();
  287.         _Nop();
  288.         _Nop();
  289.         SCL=1;       //置时钟线为高使数据线上数据有效
  290.         _Nop();
  291.         _Nop();
  292.         retc=retc<<1;
  293.         if(SDA==1)retc=retc+1; //读数据位,接收的数据位放入retc中
  294.         _Nop();
  295.         _Nop();
  296.       }
  297.   SCL=0;   
  298.   _Nop();
  299.   _Nop();
  300.   return(retc);
  301. }



  302. /*----------------------------------------------------------------
  303.                      应答子函数
  304. 原型:  void Ack_I2c(void);

  305. ----------------------------------------------------------------*/
  306. void Ack_I2c(void)
  307. {
  308.   
  309.   SDA=0;     
  310.   _Nop();
  311.   _Nop();
  312.   _Nop();      
  313.   SCL=1;
  314.   _Nop();
  315.   _Nop();              //时钟低电平周期大于4μ
  316.   _Nop();
  317.   _Nop();
  318.   _Nop();  
  319.   SCL=0;               //清时钟线,钳住I2C总线以便继续接收
  320.   _Nop();
  321.   _Nop();   
  322. }
  323. /*----------------------------------------------------------------
  324.                      非应答子函数
  325. 原型:  void NoAck_I2c(void);

  326. ----------------------------------------------------------------*/
  327. void NoAck_I2c(void)
  328. {
  329.   
  330.   SDA=1;
  331.   _Nop();
  332.   _Nop();
  333.   _Nop();      
  334.   SCL=1;
  335.   _Nop();
  336.   _Nop();              //时钟低电平周期大于4μ
  337.   _Nop();
  338.   _Nop();
  339.   _Nop();  
  340.   SCL=0;                //清时钟线,钳住I2C总线以便继续接收
  341.   _Nop();
  342.   _Nop();   
  343. }

  344. /*----------------------------------------------------------------
  345.                     向有子地址器件发送多字节数据函数               
  346. 函数原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
  347. 功能:     从启动总线到发送地址,子地址,数据,结束总线的全过程,从器件
  348.           地址sla,子地址suba,发送内容是s指向的内容,发送no个字节。
  349.            如果返回1表示操作成功,否则操作有误。
  350. 注意:    使用前必须已结束总线。
  351. ----------------------------------------------------------------*/
  352. bit ISendStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
  353. {
  354.    unsigned char i;

  355.    Start_I2c();               //启动总线
  356.    SendByte(sla);             //发送器件地址
  357.      if(ack==0)return(0);
  358.    SendByte(suba);            //发送器件子地址
  359.      if(ack==0)return(0);

  360.    for(i=0;i<no;i++)
  361.     {   
  362.      SendByte(*s);            //发送数据
  363.      Delay_ms(1);
  364.        if(ack==0)return(0);
  365.      s++;
  366.     }
  367. Stop_I2c();                  //结束总线
  368.   return(1);
  369. }

  370. /*----------------------------------------------------------------
  371.                     向有子地址器件读取多字节数据函数               
  372. 函数原型: bit  ISendStr(unsigned char sla,unsigned char suba,ucahr *s,unsigned char no);  
  373. 功能:     从启动总线到发送地址,子地址,读数据,结束总线的全过程,从器件
  374.           地址sla,子地址suba,读出的内容放入s指向的存储区,读no个字节。
  375.            如果返回1表示操作成功,否则操作有误。
  376. 注意:    使用前必须已结束总线。
  377. ----------------------------------------------------------------*/

  378. bit IRcvStr(unsigned char sla,unsigned char suba,unsigned char *s,unsigned char no)
  379. {
  380.    unsigned char i;

  381.    Start_I2c();               //启动总线
  382.    SendByte(sla);             //发送器件地址
  383.      if(ack==0)return(0);
  384.    SendByte(suba);            //发送器件子地址
  385.      if(ack==0)return(0);

  386.    Start_I2c();
  387.    SendByte(sla+1);
  388.       if(ack==0)return(0);

  389.    for(i=0;i<no-1;i++)
  390.     {   
  391.      *s=RcvByte();              //发送数据
  392.       Ack_I2c();                //发送就答位
  393.      s++;
  394.     }
  395.    *s=RcvByte();
  396.     NoAck_I2c();                 //发送非应位
  397.    Stop_I2c();                    //结束总线
  398. ……………………

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

所有资料51hei提供下载:
tempature.zip (98.2 KB, 下载次数: 62)



评分

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

查看全部评分

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

使用道具 举报

沙发
ID:460624 发表于 2019-1-3 09:56 | 只看该作者
程序好像有问题。。。
回复

使用道具 举报

板凳
ID:460624 发表于 2019-1-3 09:57 | 只看该作者
楼主用的时候有警告吗
回复

使用道具 举报

地板
ID:400732 发表于 2019-1-20 22:14 | 只看该作者
vicson 发表于 2019-1-3 09:56
程序好像有问题。。。

存储数据那里仿真做不了,应该是有问题的,谢谢批评指正
回复

使用道具 举报

5#
ID:400732 发表于 2019-1-20 22:15 | 只看该作者
vicson 发表于 2019-1-3 09:57
楼主用的时候有警告吗

我的是灯光警告呢。
回复

使用道具 举报

6#
ID:774414 发表于 2020-6-10 10:40 | 只看该作者
那个文件可以仿真出来吗
回复

使用道具 举报

7#
ID:774414 发表于 2020-6-10 22:42 | 只看该作者
小小徐白 发表于 2020-6-10 10:40
那个文件可以仿真出来吗

老是,停留在15不变
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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