|
单片机温度计分享
- #include<reg51.h> //包含单片机寄存器的头文件
- #include<intrins.h> //包含_nop_()函数定义的头文件
- #include <absacc.h>
- #include <math.h>
- unsigned char code dispcode0[]={0x3f,0x06,0x5b,0x4f,
- 0x66,0x6d,0x7d,0x07,
- 0x7f,0x6f};
- unsigned char dispcode1[]={0xbf,0x86,0xdb,0xcf,0xe6,0xed,
- 0xfd,0x87,0xff,0xef};
- unsigned char TL; //储存暂存器的温度低位
- unsigned char TH; //储存暂存器的温度高位
- unsigned char TN; //储存温度的整数部分
- unsigned int TD; //储存温度的小数部分
- void delay1ms()
- {
- unsigned char i,j;
- for(i=0;i<10;i++)
- for(j=0;j<33;j++)
- ;
- }
- void delaynms(int n)
- {
- unsigned char i;
- for(i=0;i<n;i++)
- delay1ms();
- }
- sbit DQ=P1^0;
- unsigned char time_DS18B20; //设置全局变量,专门用于严格延时
- bit Init_DS18B20(void)
- {
- bit flag_DS18B20; //储存DS18B20是否存在的标志,flag=0,表示存在;flag=1,表示不存在
- DQ = 1; //先将数据线拉高
- for(time_DS18B20=0;time_DS18B20<2;time_DS18B20++) //略微延时约6微秒
- ;
- DQ = 0; //再将数据线从高拉低,要求保持480~960us
- for(time_DS18B20=0;time_DS18B20<200;time_DS18B20++) //略微延时约600微秒
- ; //以向DS18B20发出一持续480~960us的低电平复位脉冲
- DQ = 1; //释放数据线(将数据线拉高)
- for(time_DS18B20=0;time_DS18B20<10;time_DS18B20++)
- ; //延时约30us(释放总线后需等待15~60us让DS18B20输出存在脉冲)
- flag_DS18B20=DQ; //让单片机检测是否输出了存在脉冲(DQ=0表示存在)
- for(time_DS18B20=0;time_DS18B20<200;time_DS18B20++) //延时足够长时间,等待存在脉冲输出完毕
- ;
- return (flag_DS18B20); //返回检测成功标志
- }
- unsigned char ReadOneChar( )
- {
- unsigned char i=0;
- unsigned char dat; //储存读出的一个字节数据
- for (i=0;i<8;i++)
- {
- DQ =1; // 先将数据线拉高
- _nop_(); //等待一个机器周期
- DQ = 0; //单片机从DS18B20读书据时,将数据线从高拉低即启动读时序
- dat>>=1;
- _nop_(); //等待一个机器周期
- DQ = 1; //将数据线"人为"拉高,为单片机检测DS18B20的输出电平作准备
- for(time_DS18B20=0;time_DS18B20<3;time_DS18B20++); //延时约6us,使主机在15us内采样
- if(DQ==1)
- dat|=0x80; //如果读到的数据是1,则将1存入dat
- else
- dat|=0x00;//如果读到的数据是0,则将0存入dat
- //将单片机检测到的电平信号DQ存入r[i]
- for(time_DS18B20=0;time_DS18B20<8;time_DS18B20++)
- ; //延时3us,两个读时序之间必须有大于1us的恢复期
- }
- return(dat); //返回读出的十进制数据
- }
- WriteOneChar(unsigned char dat)
- {
- unsigned char i=0;
- for (i=0; i<8; i++)
- {
- DQ =1; // 先将数据线拉高
- _nop_(); //等待一个机器周期
- DQ=0; //将数据线从高拉低时即启动写时序
- DQ=dat&0x01; //利用与运算取出要写的某位二进制数据,
- //并将其送到数据线上等待DS18B20采样
- for(time_DS18B20=0;time_DS18B20<10;time_DS18B20++)
- ;//延时约30us,DS18B20在拉低后的约15~60us期间从数据线上采样
- DQ=1; //释放数据线
- for(time_DS18B20=0;time_DS18B20<1;time_DS18B20++)
- ;//延时3us,两个写时序间至少需要1us的恢复期
- dat>>=1; //将dat中的各二进制位数据右移1位
- }
- for(time_DS18B20=0;time_DS18B20<4;time_DS18B20++)
- ; //稍作延时,给硬件一点反应时间
- }
- void display(unsigned char x,y)
- {
- unsigned char j,k,l,m,n,o; //j,k,l分别储存温度的百位、十位和个位
- j=x/100; //取百位
- k=(x%100)/10; //取十位
- l=x%10; //取个位
- m=y/100;
- n=(y%100)/10;
- o=y/10;
- if(x>=100)
- {
- P0=dispcode0[j];
- P2=dispcode0[k];
- if(o>=5)
- n+=1;
- else
- ;
- if(n>=5)
- m+=1;
- else
- ;
- if(m>=5)
- l+=1;
- else
- ;
- P3=dispcode0[l];
- }
- else if((x>=10)&&(x<100))
- {
- P0=dispcode0[k];
- P2=dispcode1[l];
- if(o>=5)
- n+=1;
- else
- ;
- if(n>=5)
- m+=1;
- else
- ;
- P3=dispcode0[m];
- }
- else if((x>=0)&&(x<10))
- {
- P0=dispcode1[l];
- P2=dispcode0[m];
- if(o>=5)
- n+=1;
- else
- ;
- P3=dispcode0[n];
- }
- }
- void displayfu(unsigned char x,y)
- {
- unsigned char k,l,m,n,o; //j,k,l分别储存温度的百位、十位和个位
- P0=0x40;
- k=(x%100)/10; //取十位
- l=x%10; //取个位
- m=y/100;
- n=(y%100)/10;
- o=y/10;
- if(x>=10)
- {
- P2=dispcode0[k];
- if(o>=5)
- n+=1;
- else
- ;
- if(n>=5)
- m+=1;
- else
- ;
- if(m>=5)
- l+=1;
- else
- ;
- P3=dispcode0[l];
- }
- else if(x<10)
- {
- P2=dispcode1[l];
- if(o>=5)
- n+=1;
- else
- ;
- if(n>=5)
- m+=1;
- else
- ;
- P3=dispcode0[m];
- }
- }
- void ReadyReadTemp(void)
- {
- Init_DS18B20(); //将DS18B20初始化
- WriteOneChar(0xCC); // 跳过读序号列号的操作
- WriteOneChar(0x44); // 启动温度转换
- for(time_DS18B20=0;time_DS18B20<100;time_DS18B20++)
- ; //温度转换需要一点时间
- Init_DS18B20(); //将DS18B20初始化
- WriteOneChar(0xCC); //跳过读序号列号的操作
- WriteOneChar(0xBE); //读取温度寄存器,前两个分别是温度的低位和高位
- }
- void main(void)
- {
- while(1) //不断检测并显示温度
- {
- ReadyReadTemp(); //读温度准备
- TL=ReadOneChar(); //先读的是温度值低位
- TH=ReadOneChar(); //接着读的是温度值高位
- if(TH>=8)
- {
- TH=~TH;
- TL=~TL;
- TL=TL+1;
- if(TL==0)
- TH+=1;
- else
- ;
- TN=TH*16+TL/16;
- TD=(TL%16)*63;
- displayfu(TN,TD);
- }
- else
- {
- TN=TH*16+TL/16; //实际温度值=(TH*256+TL)/16,即:TH*16+TL/16
- TD=(TL%16)*62;
- display(TN,TD);
- }
- delaynms(200);
- }
- }
- \032
复制代码
|
-
-
温度计.rar
20.6 KB, 下载次数: 44, 下载积分: 黑币 -5
|