#include<pic.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 bai; //整数百位
unsigned char shi; //整数十位
unsigned char ge; //整数个位
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};//显示数组0
void delayus(char x,char y)//us级的延时
{
char z; //定义Z
do { //先执行一次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 bai,char shi,char ge,char shifen,char baifen,char qianfen,char wanfen)
{
PORTB=table[bai];
PORTD=0B01111111;
delay(2);
PORTB=table[shi];
PORTD=0B10111111;
delay(2);
PORTB=table[ge]|0x80; //或上0x80显示出小数点
PORTD=0B11011111;
delay(2);
PORTB=table[shifen];
PORTD=0B11101111;
delay(2);
PORTB=table[baifen];
PORTD=0B11110111;
delay(2);
PORTB=table[qianfen];
PORTD=0B11111011;
delay(2);
PORTB=table[wanfen];
PORTD=0B111111101;
delay(2);
}
void init(void)
{
TRISC=0xf0;PORTC=1;
TRISB=0;TRISD=0;TRISE=0;ANSELH=0;ANSEL=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循环重发
else pe=0; //否则就是响应了,置0以退出循环
delayus(2,81); //延时502us
}
}
void write_byte(char val)
{
uchar i,temp;
for(i=8;i>0;i--) //循环8次构成一个字节
{
temp=val&0x01; //取出,最低位,相与取出1
DQ_LOW();
delayus(1,1); //延时15us
if(temp==1)DQ_HIGH(); //判断如果取出的是1时拉至高电平,发出去
delayus(3,3); //延时45us,如果取出的是0时,也发送去
DQ_HIGH(); //拉高至高电平
NOP();NOP(); //延时2us
val=val>>1; //右移一次以便下次取出
}
}
uchar read_byte(void)
{
uchar i,val=0;
static bit j; //静态位变量,一个状态位,不能是一个字节
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; //读取数据线的状态以得到一个状态位,进行数据处理 //所以要定义static bit j;
if(j==1)val=val|0x80; //数据处理:如果读到是1先放在最高位第1位,再利用逐个后移就构与一个字节了
delayus(1,6); //延时30us,以重复以上步骤
}
return(val); //构成1个字节后返回走
}
void get_temp(void) //01:40:26 //获取温度,器件匹配(多个温感)
{
uchar TLV,THV,num; //tem1/tem2;还有2个字节温度指令
float aaa;
unsigned long int temper; //32位整数
reset(); //复位
write_byte(0xCC); //跳过ROM
write_byte(0x44); //温度转换,需延时
for(num=100;num>0;num--) //原本这里是delay(1000)延时1秒的。可效果却是晃一下就没了,
display(bai,shi,ge,shifen,baifen,qianfen,wanfen); //所以用显示的来代替延时,显100次算得差不多就是750ms以上
reset(); //每次操作RAM之前,需复位下18B20,再匹配下
write_byte(0xCC); //跳过ROM
write_byte(0xBE); //告诉它,接下来我就要读你的温度了,读暂存器
TLV=read_byte(); //RAM有9个字节(我们只需要2个字节LSB和MSB这两个字节),它读的时候都是从最低位开始读
THV=read_byte();
DQ_HIGH(); //释放总线
temper=(THV*256+TLV)*625;
bai=temper/1000000;
shi=temper%1000000/100000;
ge=temper%100000/10000;
shifen=temper%10000/1000;
baifen=temper%1000/100;
qianfen=temper%100/10;
wanfen=temper%10;
}
void main()
{
init ();
while(1)
{
get_temp();
display(bai,shi,ge,shifen,baifen,qianfen,wanfen);
}
}
__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 bai; //整数百位
unsigned char shi; //整数十位
unsigned char ge; //整数个位
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};//显示数组0
void delayus(char x,char y)//us级的延时
{
char z; //定义Z
do { //先执行一次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 bai,char shi,char ge,char shifen,char baifen,char qianfen,char wanfen)
{
PORTB=table[bai];
PORTD=0B01111111;
delay(2);
PORTB=table[shi];
PORTD=0B10111111;
delay(2);
PORTB=table[ge]|0x80; //或上0x80显示出小数点
PORTD=0B11011111;
delay(2);
PORTB=table[shifen];
PORTD=0B11101111;
delay(2);
PORTB=table[baifen];
PORTD=0B11110111;
delay(2);
PORTB=table[qianfen];
PORTD=0B11111011;
delay(2);
PORTB=table[wanfen];
PORTD=0B111111101;
delay(2);
}
void init(void)
{
TRISC=0xf0;PORTC=1;
TRISB=0;TRISD=0;TRISE=0;ANSELH=0;ANSEL=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循环重发
else pe=0; //否则就是响应了,置0以退出循环
delayus(2,81); //延时502us
}
}
void write_byte(char val)
{
uchar i,temp;
for(i=8;i>0;i--) //循环8次构成一个字节
{
temp=val&0x01; //取出,最低位,相与取出1
DQ_LOW();
delayus(1,1); //延时15us
if(temp==1)DQ_HIGH(); //判断如果取出的是1时拉至高电平,发出去
delayus(3,3); //延时45us,如果取出的是0时,也发送去
DQ_HIGH(); //拉高至高电平
NOP();NOP(); //延时2us
val=val>>1; //右移一次以便下次取出
}
}
uchar read_byte(void)
{
uchar i,val=0;
static bit j; //静态位变量,一个状态位,不能是一个字节
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; //读取数据线的状态以得到一个状态位,进行数据处理 //所以要定义static bit j;
if(j==1)val=val|0x80; //数据处理:如果读到是1先放在最高位第1位,再利用逐个后移就构与一个字节了
delayus(1,6); //延时30us,以重复以上步骤
}
return(val); //构成1个字节后返回走
}
void get_temp(void) //01:40:26 //获取温度,器件匹配(多个温感)
{
uchar TLV,THV,num; //tem1/tem2;还有2个字节温度指令
float aaa;
unsigned long int temper; //32位整数
reset(); //复位
write_byte(0xCC); //跳过ROM
write_byte(0x44); //温度转换,需延时
for(num=100;num>0;num--) //原本这里是delay(1000)延时1秒的。可效果却是晃一下就没了,
display(bai,shi,ge,shifen,baifen,qianfen,wanfen); //所以用显示的来代替延时,显100次算得差不多就是750ms以上
reset(); //每次操作RAM之前,需复位下18B20,再匹配下
write_byte(0xCC); //跳过ROM
write_byte(0xBE); //告诉它,接下来我就要读你的温度了,读暂存器
TLV=read_byte(); //RAM有9个字节(我们只需要2个字节LSB和MSB这两个字节),它读的时候都是从最低位开始读
THV=read_byte();
DQ_HIGH(); //释放总线
temper=(THV*256+TLV)*625;
bai=temper/1000000;
shi=temper%1000000/100000;
ge=temper%100000/10000;
shifen=temper%10000/1000;
baifen=temper%1000/100;
qianfen=temper%100/10;
wanfen=temper%10;
}
void main()
{
init ();
while(1)
{
get_temp();
display(bai,shi,ge,shifen,baifen,qianfen,wanfen);
}
}