DS18B20制作的溫度測量模塊,這款能顯示正負值的單片機DS18B20測溫模塊是由電子樂屋源創(chuàng)制作,單片機驅動數(shù)碼管的端口設置成推挽工作方式,這樣使用整個顯示電路比較簡單,數(shù)碼管段驅動端省去了限流電阻,數(shù)碼管亮度顯示通過程序控制通斷時間實現(xiàn)。只使用了6只元件:一只DS18B20數(shù)字溫度傳感器、一個USB插口、一片STC12C4052單片機、一個4位一體共陽數(shù)碼管,一個10uf貼片復位電容、一個10k的貼片復位電阻。由于電路比較簡單,這里直接給出PCB圖,設計溫度測量范圍是:-9.9~99.9℃,下面是制作過程,文后附有源程序,源程序適合于STC1T單片機。下圖是制作好的實物工作照片。
為了方便單片機愛好都仿制,附上源程序:注意(貼出來的程序可能頭文件會發(fā)生變化,注意自己修改一下
//使用單片機內部RC振蕩器
#include
#include
#define uchar unsigned char
#define uint unsigned int
sfr P1M0 = 0x91;
sfr P1M1 = 0x92;
sfr P3M0 = 0xB1;
sfr P3M1 = 0xB2;
#define ENABLE_ISP 0x84 //系統(tǒng)工作時鐘<6MHz 時,對IAP_CONTR 寄存器設置此值
sbit temp=P1^7;
sbit LED0=P3^0; //C
sbit LED1=P1^4; // 小數(shù)點后一位
sbit LED2=P1^3; //個位
sbit LED3=P1^0; // 十位
sbit A=P1^1;
sbit B_B=P1^5;
sbit C=P3^2;
sbit D=P3^4;
sbit E=P3^5;
sbit F=P1^2;
sbit G=P3^1;
sbit H=P3^3; //小數(shù)點
uchar temp_low,zf,mz;
int temp_high;
int final_temp;
void dm(mz);
void delay(uint x) //(x+1)*6微
{
while(x--);
}
void delay_long(uint x)
{
uint i;
while(x--)
{
for(i=0;i<125;i++);
}
}
void init_ds18b20()//初始化
{
temp=1;//復位
delay(6);//稍作延時
temp=0;
delay(145);//延時大于480us(520us)
temp=1;
delay(14);//這個時間不能太長,否則就過了檢測信號的時間了
void read_signal()//讀取應答脈沖
{
while(temp);
while(~temp)//檢測到應答脈沖
{
delay(7);
break;
}
}
bit readbit_ds18b20()
{
bit b;
temp=1;
delay(6);//稍作延時
temp=0;
delay(2);//保持低最少1us(4us)
temp=1;
delay(4);//延時15us以后輸出數(shù)據(jù)有效(23us)
b=temp;
delay(20);//讀時間間隙不少于60us(71us)
return(b);
}
void writebyte_ds18b20(uchar b)//寫0寫1一起完成
{
int i,j;
uchar btemp;
temp=1;
for(i=0;i<8;i++)
{
j=0;
btemp=b&0x01;
b>>=1;
if(btemp==0)
{
temp=0;
delay(18);//保持拉低在60us以上(71us)
temp=1;
}
else
{
temp=0;
j++;//15us之內拉高
temp=1;
delay(18);//整個寫時序時間在60us以上(71us)
}
}
}
void temp_convert()
{
init_ds18b20();//初始化
read_signal();//讀取應答脈沖
delay_long(4);
writebyte_ds18b20(0xcc);//跳過驗證序列號命令,若單線上有多個ds18b20,則不可用這個命令
writebyte_ds18b20(0x44);//啟動溫度轉換命令
}
char readbyte_ds18b20()
{
uint i;
uchar a,b;
b=0;
for(i=0;i<8;i++)
{
a=readbit_ds18b20();
b=(a<
}
return(b);
}
uint read_ds18b20()
{
int y;
float yy;
init_ds18b20();//初始化
read_signal();//讀取應答脈沖
delay_long(4);
writebyte_ds18b20(0xcc);//跳過驗證序列號命令
writebyte_ds18b20(0xbbe);//讀取內部ROM的數(shù)據(jù)
temp_low=readbyte_ds18b20();//讀數(shù)據(jù)時低位在前,高位在后
temp_high=readbyte_ds18b20();
y=temp_high;
y<<=8;
y=y|temp_low;//整合為一個int型
yy=y*0.0625;//12位精度為0.0625
y=yy*10+0.5;
return(y);
}
void display(uint x)
{
uchar sw,gw,xs;
sw=x/100;
gw=x0/10; //個位
xs=x; //小數(shù)
if(zf==1)
{
sw=11;
}
else
{
if(sw==0)
{
sw=12;
}
}
dm(sw);
LED3=1;
delay(30);
LED3=0;
delay(10);
dm(gw);
LED2=1;
delay(30);
LED2=0;
delay(10);
dm(13);
LED2=1;
delay(10);
LED2=0;
delay(10);
dm(xs);
LED1=1;
delay(30);
LED1=0;
delay(10);
dm(10);
LED0=1;
delay(30);
LED0=0;
delay(10);
}
void dm(mz)
{
switch(mz)
{
case 0:A=0;B_B=0;C=0;D=0;E=0;F=0;G=1;H=1;break;
case 1:A=1;B_B=0;C=0;D=1;E=1;F=1;G=1;H=1;break;
case 2:A=0;B_B=0;C=1;D=0;E=0;F=1;G=0;H=1;break;
case 3:A=0;B_B=0;C=0;D=0;E=1;F=1;G=0;H=1;break;
case 4:A=1;B_B=0;C=0;D=1;E=1;F=0;G=0;H=1;break;
case 5:A=0;B_B=1;C=0;D=0;E=1;F=0;G=0;H=1;break;
case 6:A=0;B_B=1;C=0;D=0;E=0;F=0;G=0;H=1;break;
case 7:A=0;B_B=0;C=0;D=1;E=1;F=1;G=1;H=1;break;
case 8:A=0;B_B=0;C=0;D=0;E=0;F=0;G=0;H=1;break;
case 9:A=0;B_B=0;C=0;D=0;E=1;F=0;G=0;H=1;break;
case 10:A=0;B_B=1;C=1;D=0;E=0;F=0;G=1;H=1;break; //C
case 11:A=1;B_B=1;C=1;D=1;E=1;F=1;G=0;H=1;break; //-
case 12:A=1;B_B=1;C=1;D=1;E=1;F=1;G=1;H=1;break; //不顯示
case 13:A=1;B_B=1;C=1;D=1;E=1;F=1;G=1;H=0;break; //小數(shù)點
}
}
void main(void)
{
P1M0 = 0x00;
P1M1 = 0x19;
P3M0=0x00;
P3M1=0x01;
LED0=0; //C
LED1=0; // 小數(shù)點后一位
LED2=0; //個位
LED3=0; // 十位
read_ds18b20();
temp_convert();
delay_long(5);
delay_long(2000);//delay(5)就是延時555us
while(1)
{
temp_convert();
delay_long(5);
final_temp=read_ds18b20();
if(final_temp<0)
{
final_temp=-(final_temp-1);
zf=1;
}
else zf=0;
display(final_temp);
}
}