DS18B20的数字温度传感器DS18B20介绍
ds18b20温度传感器如何实现输出信号为数字信号
DS18B20本身输出的就是数字信号。你只需弄明白它的输出信号格式直接读取即可。
如何用pic单片机写DS18B20数字温度传感器的程序
#include//单总线的运用.DS18B20数字温度传感器(在I/O口上进行总线操作时,读取数据要用或运算,发送数据要用与运算) #define uchar unsigned char//宏定义 #define uint unsigned int ///这几个宏定义为了DQ 是要读和写程序所以直接宏定义可以简化设置输入输出状态 #define DQ RC1 //宏定义DQ等同于RC1这个端口 #define DQ_HIGH() TRISC1=1 //宏定义DQ高电平时设为输入状态(即DQ_HIGH()字符串等同于TRISC1=1) #define DQ_LOW() TRISC1=0;DQ=0 //宏定义DQ低电平时设为输出状态且RC1端口拉低电平(即DQ_LOW()字符串等同于TRISC1=0且RC1=0) uint temper;//先定义一个要显示温度的变量 uchar a1,a2,a3,a4;//定义数码管显示的4个变量,我们只取小数前两位和后两位 __CONFIG(0x3b31);//设置配置位 const uchar table[]={0x3f,0x06,0x5b,0x4f,//注意code是用在51单片机中的程序储存器中,const是一个常量,pic和51的单片机也可以共用的常量,但要写在前头 0x66,0x6d,0x7d,0x07, 0x7f,0x6f,0x77,0x7c, 0x39,0x5e,0x79,0x71,0x20};//数码管数字表从0,1,2,3,4,5,6,7,8,9,a,b,c,d,e,f,无显示 const uchar table1[]={0xbf,0x86,0xdb,0xcf,//带小数点的0,1,2,3,4,5,6,7,8,9 0xe6,0xed,0xfd,0x87,0xff,0xef}; void delayus(uint,uchar);//微秒的延时声明 void delay(uint x);//毫秒的延时声明 void init(); void disp(uchar num1,uchar num2,uchar num3,uchar num4); void reset(); void write_byte(uchar date); uchar read_byte(); void get_tem(); void main() { init();//调用初始化 while(1)//因为要不断地循环扫描键盘检测是否按下所以要进行死循环 { // NOP();//单片机的空指令可以当作1us延时使用,不用声明,但一定要大写 // delayus(0,0);//20us可用软件调试仿真的Stopwatch可得20us,30us,45us,70us,500us,750us // delayus(1,1);//30us // delayus(2,2);//45us // delayus(4,4);//70us // delayus(70,30);//750us // delayus(50,10);//500us get_tem();//调用获取DS18B20温度程序 // for(num=20;num>0;num--)//隔20us变更一次 // disp(a1,a2,a3,a4);//同时调用数码管 } } void reset()//DS18B20的初始化工作时序而不是单片机的 { uchar st=1;//在初始化中要读DS18B20返回的低电平,所以要先定义一个变量st,且等于1 DQ_HIGH();//上面已定义了等同于TRISC1=1即设置RC1为输入状态,又因为原理图上有上拉电阻,所以为高电平 NOP();NOP();//延时2us while(st)//循环st=0为假说明DS18B20已经返回0响应了确定存在,退出while { DQ_LOW();//上面已定义了等同于TRISC1=0,RC1=0即设置RC1为输出状态,且输出低电平 delayus(70,30);//延时750us DQ_HIGH();//拉到高电平 delayus(4,4);//延时40us if(DQ==1)//进行判断如果等于1,则at=1,DS18B20没有返回低电平未有响应 st=1;//等于1则要超过或循环while语句重新发送给DS18B20响应,不可能一次就确定18b20的存在 else st=0;//循环直到st=0为假说明DS18B20已经返回0响应了 delayus(50,10);//因为已经有返回响应,确定DS18B20的存在,所以要延时500us再退出while } DQ_HIGH();//重新拉高,也叫释放总线 } void write_byte(uchar date)//DS18B20的写工作时序,里面的date是单片机要发送的数据 { uchar i,temp;//定义一个for循环的变量和发送数据中的一个位的变量 DQ_HIGH();//先置高电平 NOP();NOP();//延时2us for(i=8;i>0;i--)//因为发送一个数据有8位 { temp=date&0x01;//和00000001与,无论date是什么数与之后只有最低位是有效的,temp得到的其实是date的最低的一位 DQ_LOW();//置低电平 delayus(0,0);//延时20us if(temp==1)//说明date的最低位是1,用if,else语句把数据从最低位到高一位一位的发送 DQ_HIGH();//因为temp=1表示数据线要置高电平 else DQ_LOW();//表示temp=0数据线要置低电平 delayus(2,2);//延时45us DQ_HIGH();//重新拉高,也叫释放总线 date=date>>1;//发送完一位后需要把date右移一位才能进行循环,如原来是01010101,右移1位后得到00101010,最低位被移走即发送 } } uchar read_byte()//DS18B20的读工作时序,因为是读所以是一个带返回值的函数,括号里面不用写变量,因为单片机只是读取而不发送任何东西 { uchar i,date;//再定义一个for循环的变量i和接收数据的变量date static bit j;//定义一个状态位,j是一个位的变量 for(i=8;i>0;i--)//因为接收一个数据有8位 { date=date>>1;//先将数据右移一位其实这里只移动7位,加上或运算移动一位就共8位 DQ_HIGH();//先要确定数据线拉高 NOP();NOP();//延时2us DQ_LOW();//将数据线拉低 NOP();NOP();NOP();NOP();NOP();NOP();//延时6us DQ_HIGH();//拉高 NOP();NOP();NOP();NOP();//延时4us j=DQ;//把RC1数据线的状态附给状态位j,这样读取到的数据线高低电平就是j的变化 if(j==1)//如果等于1,则说明是高电平,等于0时不需要或运算,因为或运算相当于右移,最高位自动补0 date=date|0x80;//只有读回来的数是1时才和10000000或运算,因为第一个读回来的是最低位,如果第二个又读回到要放在倒数第二位会不好放,所以要将最低位或运算放在最高位,这里已经移动过一次了 //如date是1或运算后得10000000,而这里只读取一次,循环后可得第二个11000000如果是0则直接填10000000 delayus(1,1);//延时30us } return (date);//把接收到的数据返回去经单片机 } void get_tem()//获取温度指令将数据化为温度给数码管显示的函数 { uchar temp1,temp2,num;//因为同时一次从低到高读两个字节,定义两个字节的变量,是下面的指令的变量 float aaa;//定义一个浮点数等于aaa的变量,提高精确度 reset();//调用DS18B20初始化相当复位 write_byte(0xcc);//ccH,因为只接了一个不需要配对,跳过了匹配的ROM指令 write_byte(0x44);//发送温度转换指令44H for(num=100;num>0;num--)//隔100次,数码管闪一次 disp(a1,a2,a3,a4);//同时调用数码管 reset();//重新复位 write_byte(0xcc);//ccH,因为只接了一个不需要配对,跳过了匹配的ROM指令 write_byte(0xbe);//BEH是指接下来我要读你的指令 temp1=read_byte(); temp2=read_byte();//因为同时一次从低到高读两个字节 // temper=(temp2*256+temp1)*0.0625*100;//将温度转换成十六位温度数据,转换成十进制还需要乘以0.0625,因为我们只显示4个数码管,后两个是小数,不好提取就乘以100变成整数再提取 aaa=(temp2*256+temp1)*0.0625*100;//因为前面定义temper是一个整形的变量,乘出来的会是取整数精确度不高,附给用浮点数float表示的aaa就可以乘出小数部分 temper=(int)aaa;//再将aaa强制转换给整形temper,这时的整形temper就可以是带小数的了,注意书写格式 //这里面是强制转换的指令 a1=temper/1000;//因上一条程序已化为4位整数,提取对最高位千位求模 a2=temper%1000/100;//提取对百位求模 a3=temper%100/10;//提取对十位求模 a4=temper%10;//提取对个位求佘 } void delayus(uint x,uchar y)//定义一个整形一个字符形变量表示微秒 { uint i; uchar j; for(i=x;i>0;i--); for(j=y;j>0;j--); } void delay(uint x)//延迟函数x表示毫秒 { uint a,b; for(a=x;a>0;a--) for(b=110;b>0;b--);//嵌套 } void init() { TRISD=0;//因为RD接的是数码管设置全为输出状态 TRISA=0;//设置数码管的位选为全输出状态 PORTD=0;//设置输出先全部关闭 PORTA=0;//在初始化时数码管不能显示 } void disp(uchar num1,uchar num2,uchar num3,uchar num4)//数码管的扫描函数,要在里面有4个变量,每一个为一个数码管显示的数 { PORTD=table[num1];//调用数码管的显示函数(注第一个是显示0)这是从左到右第一个数码管要显示的段选 PORTA=0x20;//00100000由原理图可得第一个数码管是由RA5控制位选的 delay(10);//因为是要动态,所以要加延时,但时间不能太长 PORTD=table1[num2];//调用数码管的显示函数(注第一个是显示0)这是第二个数码管要显示的段选,显示的小带小数点的 PORTA=0x10;//00010000由原理图可得第二个数码管是由RA4控制位选的 delay(10);//因为是要动态,所以要加延时,但时间不能太长 PORTD=table[num3];//调用数码管的显示函数(注第一个是显示0)这是第三个数码管要显示的段选 PORTA=0x08;//00001000由原理图可得第三个数码管是由RA3控制位选的 delay(10);//因为是要动态,所以要加延时,但时间不能太长 PORTD=table[num4];//调用数码管的显示函数(注第一个是显示0)这是第四个数码管要显示的段选 PORTA=0x04;//00000100由原理图可得第四个数码管是由RA2控制位选的 delay(10);//因为是要动态,所以要加延时,但时间不能太长 }
DS18B20与单片机结合来测量温度。利用数字温度传感器DS18B20测量温度信号
#include #define uchar unsigned char sbit keyup=P1^0; sbit keydn=P1^1; sbit keymd=P1^2; sbit out=P3^7; //接控制继电器 sbit DQ = P3^4; //接温度传感器18B20 uchar t[2],number=0,*pt; //温度值 uchar TempBuffer1[4]={0,0,0,0}; uchar Tmax=18,Tmin=8; uchar distab[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff,0xfe,0xf7}; uchar dismod=0,xiaodou1=0,xiaodou2=0,currtemp; bit flag; void t0isr() interrupt 1 { TH0=(65536-5000)/256; TL0=(65536-5000)%256; switch(number) { case 0: P2=0x08; P0=distab[TempBuffer1[0]]; break; case 1: P2=0x04; P0=distab[TempBuffer1[1]]; break; case 2: P2=0x02; P0=distab[TempBuffer1[2]]&0x7f; break; case 3: P2=0x01; P0=distab[TempBuffer1[3]]; break; default: break; } number++; if(number>3)number=0; } void delay_18B20(unsigned int i) { while(i--); } /**********ds18b20初始化函数**********************/ void Init_DS18B20(void) { bit x=0; do{ DQ=1; delay_18B20(8); DQ = 0; //单片机将DQ拉低 delay_18B20(90); //精确延时 大于 480us DQ = 1; //拉高总线 delay_18B20(14); x=DQ; //稍做延时后 如果x=0则初始化成功 x=1则初始化失败,继续初始化 }while(x); delay_18B20(20); } /***********ds18b20读一个字节**************/ unsigned char ReadOneChar(void) { unsigned char i=0; unsigned char dat = 0; for (i=8;i>0;i--) { DQ = 0; // 给脉冲信号 dat>>=1; DQ = 1; // 给脉冲信号 if(DQ) dat|=0x80; delay_18B20(4); } return(dat); } /*************ds18b20写一个字节****************/ void WriteOneChar(unsigned char dat) { unsigned char i=0; for (i=8; i>0; i--) { DQ = 0; DQ = dat&0x01; delay_18B20(5); DQ = 1; dat>>=1; } } /**************读取ds18b20当前温度************/ unsigned char *ReadTemperature(unsigned char rs) { unsigned char tt[2]; delay_18B20(80); Init_DS18B20(); WriteOneChar(0xCC); //跳过读序号列号的操作 WriteOneChar(0x44); //启动温度转换 delay_18B20(80); Init_DS18B20(); WriteOneChar(0xCC); //跳过读序号列号的操作 WriteOneChar(0xBE); //读取温度寄存器等(共可读9个寄存器)前两个就是温度 tt[0]=ReadOneChar(); //读取温度值低位 tt[1]=ReadOneChar(); //读取温度值高位 return(tt); } void covert1(void) //将温度转换为LED显示的数据 { uchar x=0x00,y=0x00; t[0]=*pt; pt++; t[1]=*pt; if(t[1]&0x080) //判断正负温度 { TempBuffer1[0]=0x0c; //c代表负 t[1]=~t[1]; /*下面几句把负数的补码*/ t[0]=~t[0]; /*换算成绝对值*********/ x=t[0]+1; t[0]=x; if(x==0x00)t[1]++; } else TempBuffer1[0]=0x0a; //A代表正 t[1]<<=4; //将高字节左移4位 t[1]=t[1]&0xf0; x=t[0]; //将t[0]暂存到X,因为取小数部分还要用到它 x>>=4; //右移4位 x=x&0x0f; //和前面两句就是取出t[0]的高四位 y=t[1]|x; //将高低字节的有效值的整数部分拼成一个字节 TempBuffer1[1]=(y%100)/10; TempBuffer1[2]=(y%100)%10; t[0]=t[0]&0x0f; //小数部分 TempBuffer1[3]=t[0]*10/16; //以下程序段消去随机误检查造成的误判,只有连续12次检测到温度超出限制才切换加热装置 if(currtemp>Tmin)xiaodou1=0; if(y<Tmin) { xiaodou1++; currtemp=y; xiaodou2=0; } if(xiaodou1>12) { out=0; flag=1; xiaodou1=0; } if(currtemp<Tmax)xiaodou2=0; if(y>Tmax) { xiaodou2++; currtemp=y; xiaodou1=0; } if(xiaodou2>12) { out=1; flag=0; xiaodou2=0; } out=flag; } void convert(char tmp) { uchar a; if(tmp<0) { TempBuffer1[0]=0x0c; a=~tmp+1; } else { TempBuffer1[0]=0x0a; a=tmp; } TempBuffer1[1]=(a%100)/10; TempBuffer1[2]=(a%100)%10; } void keyscan( ) { uchar keyin; keyin=P1&0x07; if(keyin==0x07)return; else if(keymd==0) { dismod++; dismod%=3; while(keymd==0); switch(dismod) { case 1: convert(Tmax); TempBuffer1[3]=0x11; break; case 2: convert(Tmin); TempBuffer1[3]=0x12; break; default: break; } } else if((keyup==0)&&(dismod==1)) { Tmax++; convert(Tmax); while(keyup==0); } else if((keydn==0)&&(dismod==1)) { Tmax--; convert(Tmax); while(keydn==0); } else if((keyup==0)&&(dismod==2)) { Tmin++; convert(Tmin); while(keyup==0); } else if((keydn==0)&&(dismod==2)) { Tmin--; convert(Tmin); while(keydn==0); } xiaodou1=0; xiaodou2=0; } main() { TMOD=0x01; TH0=(65536-5000)/256; TL0=(65536-5000)%256; TR0=1; ET0=1; EA=1; out=1; flag=0; ReadTemperature(0x3f); delay_18B20(50000); //延时等待18B20数据稳定 while(1) { pt=ReadTemperature(0x7f); //读取温度,温度值存放在一个两个字节的数组中 if(dismod==0)covert1(); keyscan(); delay_18B20(30000); } }
ds18b20数字温度传感器与传统热敏电阻有什么不同
ds18b20是单总线数字输出,读取结果需要MCU按照极其严格的时序获取结果。 热敏电阻,温度不同,阻值不同,模拟量输出。通过ADC,获取到阻值进而通过查表得到温度数值。