本帖最后由 yesno 于 2012-5-19 15:29 编辑
#include<reg52.h>
#include<intrins.h>
#define uchar unsigned char
#define uint unsigned int
sbit dula=P2^6;
sbit wela=P2^7;
sbit DQ=P2^2;
uint temp;
uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71}; //0-9数字
void delay (uint x) //延时函数
{
uchar i;
while(x--)
for(i=0; i<120; i++);
}
void delay1(uint a) //延时函数
{
while(--a);
}
void disp(uchar num) //显示函数
{
uchar shi,ge;
shi=num/10;
ge=num%10;
dula=1;
P0=table[shi];
dula=0;
P0=0xff;
wela=1;
P0=0xfe;
wela=0;
delay(5);
dula=1;
P0=table[ge];
dula=0;
P0=0xff;
wela=1;
P0=0xfd;
wela=0;
delay(5);
}
void init_18b20() // 初始化
{
DQ=1;
delay1(8);
DQ=0;
delay1(90);
DQ=1;
_nop_();
_nop_();
delay1(100);
DQ=1;
}
void write(uchar dat) //写字节
{
uchar i;
for(i=0; i<8; i++)
{
DQ=0;
DQ=dat&0x01;
delay1(5);
DQ=1;
dat>>=1;
}
delay1(4);
}
uchar read() //读字节
{
uchar i, dat=0;
DQ=1;
_nop_();
for(i=0; i<8; i++)
{
DQ=0;
_nop_();
_nop_();
dat>>=1;
DQ=1;
_nop_();
_nop_();
if(DQ)
dat|=0x80;
delay1(30);
DQ=1;
}
return dat;
}
uchar read_tu() // 温度转换
{
uchar a,b;
init_18b20();
delay1(100);
write(0xcc);
write(0x44);
init_18b20();
write(0xcc);
write(0xbe);
a=read();
b=read();
b<<=4;
b+=(a&0xf0)>>4;
return b;
}
void main() //主函数
{
delay(30);
while(1)
{
temp=read_tu;
disp(temp);
}
}
这个初学仔细检查啦视乎没错,可是下载到板子里面就不行啦。其他的情况也没什么就是温度没反应,数码管就是显示70,任凭我怎么触摸传感器都是没反应既不升温也不降温。
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
rar在上面啊
测试的怎么样啦
51单片机是8位的系统,当一个变量用32位定义时,编译器会对该变量做一系列寄存器操作。
你期望延时100us,但实际上会长很多。
//**********************************************************************************
//延时函数, 对于12MHz时钟, 例z=10,则大概延时10ms.
void delay(uint z)
{
uint x,y;
for(x=z;x>0;x--);
for(y=124;y>0;y--);
}
//**********************************************************************************
//**********************************************************************************
//初始化DS18B20
//让DS18B20一段相对长时间低电平, 然后一段相对非常短时间高电平, 即可启动
void dsreset(void)
{ //对于11.0592MHz时钟, unsigned int型的i, 作一个i++操作的时间大于?us
uint i;
ds=0;
i=103; //拉低约800us, 符合协议要求的480us以上
while(i>0)i--;
ds=1; //产生一个上升沿, 进入等待应答状态
i=4;
while(i>0)i--;
}
//**********************************************************************************
//**********************************************************************************
//向DS18B20读取一位数据
//读一位, 让DS18B20一小周期低电平, 然后两小周期高电平,
//之后DS18B20则会输出持续一段时间的一位数据
bit tempreadbit(void)
{
uint i;
bit dat;
ds=0;
i++; //i++起延迟作用 延时约8us, 符合协议要求至少保持1us
ds=1;
i++;i++; //延时约16us, 符合协议要求的至少延时15us以上
dat=ds;
i=8;
while(i>0)i--; //延时约64us, 符合读时隙不低于60us要求
return(dat);
}
//**********************************************************************************
//**********************************************************************************
//读取一字节数据, 通过调用tempreadbit()来实现
uchar tempread(void)
{
uchar i,j,dat;
dat=0;
for(i=1;i<=8;i++)
{
j=tempreadbit();
dat=(j<<7)|(dat>>1); //读出的数据最低位在最前面,这样刚好一个字节在dat里
}
return(dat);
}
//**********************************************************************************
//**********************************************************************************
//向DS18B20写入一字节数据
void tempwritebyte(uchar dat)
{
uint i;
uchar j;
bit testb;
for(j=1;j<=8;j++)
{
testb=dat&0x01;
dat=dat>>1;
if(testb) //写"1", 将ds拉低15us后, 在15us~60us内将ds拉高, 即完成写1
{
ds=0;
i++;i++; //拉低约16us, 符号要求15~60us内
ds=1;
i=8;while(i>0)i--; //延时约64us, 符合写时隙不低于60us要求
}
else //写"0", 将ds拉低60us~120us
{
ds=0;
i=8;while(i>0)i--; //拉低约64us, 符号要求
ds=1;
i++;i++; //整个写0时隙过程已经超过60us, 这里就不用像写1那样, 再延时64us了
}
}
}
//**********************************************************************************
//**********************************************************************************
//向DS18B20发送温度转换命令
void tempchange(void)
{
dsreset(); //初始化DS18B20, 无论什么命令, 首先都要发起初始化
delay(1); //延时1ms, 因为DS18B20会拉低ds 60~240us作为应答信号
tempwritebyte(0xcc); //写跳过读ROM指令
tempwritebyte(0x44); //写温度转换指令
}
//**********************************************************************************
//**********************************************************************************
//向DS18B20发送读取数据命令 获取当前温度值
uint tmp()
{
float tt;
// int tt;
uchar low,high;
dsreset();
delay(1);
tempwritebyte(0xcc); //写入跳过序列号命令字 Skip Rom
tempwritebyte(0xbe); //写入读取数据令字 Read Scratchpad
//连续读取两个字节数据
low =tempread(); //读低8位
high=tempread(); //读高8位
temp=high; //高8位赋予temp
temp<<=8; //将高低两个字节合成一个整形变量 temp为16位 高8位左移8位 低8位补0
temp=temp|low; //高低8位相或=temp为16位=高8位+低8位
xs=low&0x0f; //低4位为小数部分
tt=temp>>4; //整数部分
// temp=tt*100+xs*100/16;
temp=tt*10+((((xs*5)>>2)+1)>>1);
// **************************************
//y=(((x*5)>>2)+1)>>1;
//等效y=((x*5)/4+1)/2;
//再等效y=((x*20)/16+1)/2;
//y=2*(x*10/16+0.5)/2;
//实际上y=x*10/16+0.5; //+0.5为 四舍五入
// **********************************************
// tt=temp*0.0625; //DS18B20的默认分辨率12位, 精确度为0.0625度, 即读回数据的最低位代表0.0625度
// temp = tt * 100 + (temp > 0 ? 0.5 : -0.5); //大于0加0.5, 小于0减0.5
// temp=tt*100; //将它放大100倍, 使显示时可显示小数点后两位(t=11.0625,计算得temp = 1106, 即11.06 度)
return temp; //temp是整型
}
//**********************************************************************************
//**********************************************************************************
//显示函数
void display(uint t)
{
unsigned char a1=0,a2=0,a3=0,a4=0; //临时变量
static int k=0;
a1=t/1000; //取t的千位//
a2=t%1000/100;//取t的百位//
a3=t%100/10; //取t的十位//
a4=t%10; //取t的个位//
if (k==0){A1=1;A2=0;A3=0;A4=0;P0=yima[a1];}
else if(k==1){A1=0;A2=1;A3=0;A4=0;P0=yima[a2];}
else if(k==2){A1=0;A2=0;A3=1;A4=0;P0=yima[a3];}
else if(k==3){A1=0;A2=0;A3=0;A4=1;P0=yima[a4];}
k++;
if(k>3) k=0;
}
官网提供的
一周热门 更多>