找回密码
 立即注册

QQ登录

只需一步,快速开始

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

单片机温度计原理图和仿真程序分享 带详细注释

[复制链接]
跳转到指定楼层
楼主
单片机温度计分享



  1. #include<reg51.h>    //包含单片机寄存器的头文件
  2. #include<intrins.h>  //包含_nop_()函数定义的头文件
  3. #include <absacc.h>
  4. #include <math.h>
  5. unsigned char code dispcode0[]={0x3f,0x06,0x5b,0x4f,
  6.                                0x66,0x6d,0x7d,0x07,
  7.                                0x7f,0x6f};
  8. unsigned char dispcode1[]={0xbf,0x86,0xdb,0xcf,0xe6,0xed,
  9.                                  0xfd,0x87,0xff,0xef};

  10.      unsigned char TL;     //储存暂存器的温度低位
  11.      unsigned char TH;    //储存暂存器的温度高位
  12.      unsigned char TN;      //储存温度的整数部分
  13.      unsigned int TD;       //储存温度的小数部分
  14. void delay1ms()
  15. {
  16.    unsigned char i,j;
  17.          for(i=0;i<10;i++)
  18.           for(j=0;j<33;j++)
  19.            ;
  20. }
  21. void delaynms(int n)
  22. {
  23.    unsigned char i;
  24.         for(i=0;i<n;i++)
  25.            delay1ms();
  26. }
  27. sbit DQ=P1^0;
  28. unsigned char time_DS18B20;   //设置全局变量,专门用于严格延时
  29. bit Init_DS18B20(void)
  30. {
  31. bit flag_DS18B20;         //储存DS18B20是否存在的标志,flag=0,表示存在;flag=1,表示不存在
  32. DQ = 1;           //先将数据线拉高
  33. for(time_DS18B20=0;time_DS18B20<2;time_DS18B20++) //略微延时约6微秒
  34.      ;
  35. DQ = 0;           //再将数据线从高拉低,要求保持480~960us
  36. for(time_DS18B20=0;time_DS18B20<200;time_DS18B20++)  //略微延时约600微秒
  37.      ;         //以向DS18B20发出一持续480~960us的低电平复位脉冲
  38. DQ = 1;           //释放数据线(将数据线拉高)
  39.   for(time_DS18B20=0;time_DS18B20<10;time_DS18B20++)
  40.      ;  //延时约30us(释放总线后需等待15~60us让DS18B20输出存在脉冲)
  41. flag_DS18B20=DQ;          //让单片机检测是否输出了存在脉冲(DQ=0表示存在)
  42. for(time_DS18B20=0;time_DS18B20<200;time_DS18B20++)  //延时足够长时间,等待存在脉冲输出完毕
  43.       ;
  44. return (flag_DS18B20);    //返回检测成功标志
  45. }
  46. unsigned char ReadOneChar(  )
  47. {
  48.                 unsigned char i=0;
  49.                 unsigned char dat;  //储存读出的一个字节数据
  50.                 for (i=0;i<8;i++)
  51.                  {

  52.                    DQ =1;       // 先将数据线拉高
  53.                    _nop_();            //等待一个机器周期
  54.                    DQ = 0;      //单片机从DS18B20读书据时,将数据线从高拉低即启动读时序
  55.                           dat>>=1;
  56.                    _nop_();     //等待一个机器周期
  57.                    DQ = 1;     //将数据线"人为"拉高,为单片机检测DS18B20的输出电平作准备
  58.                    for(time_DS18B20=0;time_DS18B20<3;time_DS18B20++);      //延时约6us,使主机在15us内采样
  59.                    if(DQ==1)
  60.                       dat|=0x80;  //如果读到的数据是1,则将1存入dat
  61.                         else
  62.                                 dat|=0x00;//如果读到的数据是0,则将0存入dat

  63.                      //将单片机检测到的电平信号DQ存入r[i]
  64.                    for(time_DS18B20=0;time_DS18B20<8;time_DS18B20++)
  65.                               ;              //延时3us,两个读时序之间必须有大于1us的恢复期
  66.             }
  67.          return(dat);    //返回读出的十进制数据
  68. }
  69. WriteOneChar(unsigned char dat)
  70. {
  71.         unsigned char i=0;
  72.         for (i=0; i<8; i++)
  73.                  {
  74.                   DQ =1;         // 先将数据线拉高
  75.                   _nop_();             //等待一个机器周期
  76.                   DQ=0;          //将数据线从高拉低时即启动写时序
  77.                   DQ=dat&0x01;   //利用与运算取出要写的某位二进制数据,
  78.                        //并将其送到数据线上等待DS18B20采样
  79.                  for(time_DS18B20=0;time_DS18B20<10;time_DS18B20++)
  80.                      ;//延时约30us,DS18B20在拉低后的约15~60us期间从数据线上采样
  81.                   DQ=1;          //释放数据线
  82.                   for(time_DS18B20=0;time_DS18B20<1;time_DS18B20++)
  83.                           ;//延时3us,两个写时序间至少需要1us的恢复期
  84.                   dat>>=1;       //将dat中的各二进制位数据右移1位
  85.                  }
  86.           for(time_DS18B20=0;time_DS18B20<4;time_DS18B20++)
  87.                       ; //稍作延时,给硬件一点反应时间
  88. }
  89. void display(unsigned char x,y)
  90. {
  91.         unsigned char j,k,l,m,n,o;     //j,k,l分别储存温度的百位、十位和个位
  92.         j=x/100;              //取百位
  93.         k=(x%100)/10;    //取十位
  94.         l=x%10;             //取个位
  95.         m=y/100;
  96.         n=(y%100)/10;
  97.         o=y/10;
  98.         if(x>=100)
  99.         {
  100.         P0=dispcode0[j];
  101.         P2=dispcode0[k];
  102.         if(o>=5)
  103.         n+=1;
  104.         else
  105.         ;
  106.         if(n>=5)
  107.         m+=1;
  108.         else
  109.         ;
  110.         if(m>=5)
  111.         l+=1;
  112.         else
  113.         ;
  114.         P3=dispcode0[l];
  115.         }
  116.         else if((x>=10)&&(x<100))
  117.         {
  118.         P0=dispcode0[k];
  119.         P2=dispcode1[l];
  120.         if(o>=5)
  121.         n+=1;
  122.         else
  123.         ;
  124.         if(n>=5)
  125.         m+=1;
  126.         else
  127.         ;
  128.         P3=dispcode0[m];
  129.                   }
  130.           else if((x>=0)&&(x<10))
  131.           {
  132.             P0=dispcode1[l];
  133.             P2=dispcode0[m];
  134.             if(o>=5)
  135.             n+=1;
  136.             else
  137.             ;
  138.             P3=dispcode0[n];
  139.            }


  140. }
  141. void displayfu(unsigned char x,y)
  142. {
  143.         unsigned char k,l,m,n,o;     //j,k,l分别储存温度的百位、十位和个位
  144.         P0=0x40;
  145.         k=(x%100)/10;    //取十位
  146.         l=x%10;             //取个位
  147.         m=y/100;
  148.         n=(y%100)/10;
  149.         o=y/10;
  150.         if(x>=10)
  151.         {
  152.         P2=dispcode0[k];
  153.         if(o>=5)
  154.         n+=1;
  155.         else
  156.         ;
  157.         if(n>=5)
  158.         m+=1;
  159.         else
  160.         ;
  161.         if(m>=5)
  162.         l+=1;
  163.         else
  164.         ;
  165.         P3=dispcode0[l];
  166.         }
  167.         else if(x<10)
  168.         {
  169.         P2=dispcode1[l];
  170.         if(o>=5)
  171.         n+=1;
  172.         else
  173.         ;
  174.         if(n>=5)
  175.         m+=1;
  176.         else
  177.         ;
  178.         P3=dispcode0[m];
  179.         }


  180. }





  181. void ReadyReadTemp(void)
  182. {
  183.       Init_DS18B20();     //将DS18B20初始化
  184.                 WriteOneChar(0xCC); // 跳过读序号列号的操作
  185.                 WriteOneChar(0x44); // 启动温度转换
  186.                 for(time_DS18B20=0;time_DS18B20<100;time_DS18B20++)
  187.                     ;         //温度转换需要一点时间
  188.                 Init_DS18B20();     //将DS18B20初始化
  189.                 WriteOneChar(0xCC); //跳过读序号列号的操作
  190.                 WriteOneChar(0xBE); //读取温度寄存器,前两个分别是温度的低位和高位
  191. }

  192. void main(void)
  193. {


  194.    while(1)                //不断检测并显示温度
  195.          {
  196.                 ReadyReadTemp();     //读温度准备
  197.                 TL=ReadOneChar();    //先读的是温度值低位
  198.                 TH=ReadOneChar();    //接着读的是温度值高位
  199.                 if(TH>=8)
  200.                 {
  201.                  TH=~TH;
  202.                  TL=~TL;
  203.                  TL=TL+1;
  204.                  if(TL==0)
  205.                  TH+=1;
  206.                  else
  207.                  ;
  208.                  TN=TH*16+TL/16;
  209.                  TD=(TL%16)*63;
  210.                  displayfu(TN,TD);
  211.                  }
  212.                  else
  213.                  {
  214.                  TN=TH*16+TL/16;      //实际温度值=(TH*256+TL)/16,即:TH*16+TL/16
  215.                  TD=(TL%16)*62;
  216.                  display(TN,TD);
  217.                  }
  218.                  delaynms(200);
  219.                  }
  220.                  }
  221. \032
复制代码


温度计.rar

20.6 KB, 下载次数: 44, 下载积分: 黑币 -5

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

使用道具 举报

沙发
ID:215990 发表于 2017-6-30 06:58 | 只看该作者
为啥我的仿真一直出错啊 显示第一行出错
回复

使用道具 举报

板凳
ID:322756 发表于 2018-5-5 14:02 | 只看该作者
你微笑时好美 发表于 2017-6-30 06:58
为啥我的仿真一直出错啊 显示第一行出错

可能是你的符号是中文符号,识别不了
回复

使用道具 举报

地板
ID:262176 发表于 2018-5-5 22:10 | 只看该作者
可以可以,很不错
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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