PIC單片機:溫感器ds18B20 C語言代碼
/*溫感器
采用天祥溫度轉換模塊算法
程序完整版本的下載地址:
http://www.51hei.com/f/ds1820pic.rar
*/
#includepic.h>
__CONFIG(0X3B31);
#define uint unsigned int
#define uchar unsigned char
#define DQ RC1
#define DQ_DIR TRISC1
#define DQ_HIGH() TRISC1=1
#define DQ_LOW() DQ=0;TRISC1=0
unsigned char shi; //整數(shù)十位
unsigned char ge; //整數(shù)個位
unsigned char shifen; //十分位
unsigned char baifen; //百分位
unsigned char qianfen; //千分位
unsigned char wanfen; //萬分位
const uchar table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};//顯示數(shù)組0
void delayus(char x,char y) //慧凈提供的us級的延時
{
char z; //定義Z
do { //先執(zhí)行一次do
z=y; //把Y的值給Z
do{;}while(--z); //do空語句,等待--z,直到z=0結束,do--while語句,延時在這產生
}
while (--x); //同理x在--,再產生延時
}
void delay(uint x)
{
uint a,b;
for(a=x;a>0;a--)
for(b=110;b>0;b--);
}
void display(char shi,char ge,char shifen,char baifen,char qianfen)
{
PORTD=table[shi];
PORTA=0x02;
delay(2);
PORTD=table[ge]|0x80; //或上0x80顯示出小數(shù)點
PORTA=0x04;
delay(2);
PORTD=table[shifen];
PORTA=0x06;
delay(2);
PORTD=table[baifen];
PORTA=0x08;
delay(2);
PORTD=table[qianfen];
PORTA=0x0a;
delay(2);
}
void init(void)
{
TRISA=0;PORTA=0x00;
TRISC=0xf0;PORTC=1;
TRISD=0;
}
void reset(void)
{
char pe=1;
while(pe)
{
DQ_LOW();
delayus(2,81); //延時502us
DQ_HIGH();
delayus(4,4); //延時71us
if(DQ==1)pe=1; //判斷是否響應(響應時拉至低電平),沒響應時置1循環(huán)重發(fā)
else pe=0; //否則就是響應了,置0以退出循環(huán)
delayus(2,81); //延時502us
}
}
void write_byte(char val)
{
uchar i,temp;
for(i=8;i>0;i--) //循環(huán)8次構成一個字節(jié)
{
temp=val0x01; //取出,最低位,相與取出1
DQ_LOW();
delayus(1,1); //延時15us
if(temp==1)DQ_HIGH(); //判斷如果取出的是1時拉至高電平,發(fā)出去
delayus(3,3); //延時45us,如果取出的是0時,也發(fā)送去
DQ_HIGH(); //拉高至高電平
NOP();NOP(); //延時2us
val=val>>1; //右移一次以便下次取出
}
}
uchar read_byte(void)
{
uchar i,val=0;
static bit j; //靜態(tài)位變量,一個狀態(tài)位,不能是一個字節(jié)
for(i=8;i>0;i--)
{
val=val>>1; //先移一個位
DQ_LOW(); //拉至低電平
NOP();NOP();NOP();NOP();NOP();NOP(); //延時6us
DQ_HIGH();
NOP();NOP();NOP();NOP(); //延時4us
j=DQ; //讀取數(shù)據線的狀態(tài)以得到一個狀態(tài)位,進行數(shù)據處理 //所以要定義static bit j;
if(j==1)val=val|0x80; //數(shù)據處理:如果讀到是1先放在最高位第1位,再利用逐個后移就構與一個字節(jié)了
delayus(1,6); //延時30us,以重復以上步驟
}
return(val); //構成1個字節(jié)后返回走
}
void get_temp(void) //01:40:26 //獲取溫度,器件匹配(多個溫感)
{
uchar TLV,THV,num; //tem1/tem2;還有2個字節(jié)溫度指令
float aaa;
uint temper;
reset(); //復位
write_byte(0xCC); //跳過ROM
write_byte(0x44); //溫度轉換,需延時
for(num=100;num>0;num--) //原本這里是delay(1000)延時1秒的??尚Ч麉s是晃一下就沒了,
display(shi,ge,shifen,baifen,qianfen); //所以用顯示的來代替延時,顯100次算得差不多就是750ms以上
reset(); //每次操作RAM之前,需復位下18B20,再匹配下
write_byte(0xCC); //跳過ROM
write_byte(0xBE); //告訴它,接下來我就要讀你的溫度了,讀暫存器
TLV=read_byte(); //RAM有9個字節(jié)(我們只需要2個字節(jié)LSB和MSB這兩個字節(jié)),它讀的時候都是從最低位開始讀
THV=read_byte(); //01:45:10處+銳志在58:52分處
DQ_HIGH(); //釋放總線
aaa=(THV*256+TLV)*0.0625*1000; //(16位的溫度數(shù)據)*0.0625就是實際的溫度(十進制數(shù))
temper=(int)aaa; //因為編譯時警告;有小數(shù)點是浮點類到整形轉換;我們用強制轉換成整形 要把小數(shù)點的值取出來(小數(shù)點不好取,用乘以100來?。?br />shi=temper/10000; //幾十點幾幾分配到五個數(shù)碼管上,四個數(shù)碼顯示的感覺怪怪的所以用五個數(shù)碼管;在1:51:00處講
ge=temper%10000/1000; //我是要用五個數(shù)碼管來顯示所以就是10000五位數(shù)
shifen=temper%1000/100; //
baifen=temper%100/10; //
qianfen=temper%10; //
}
void main()
{
init ();
while(1)
{
get_temp();
display(shi,ge,shifen,baifen,qianfen); //沒帶形參編譯器過不了,帶類形的形參也過不了
}
}
評論