求救关于C51单片机ds18b20温度显示的问题

先附仿真图!程序如下#include <reg51.h>
#include <intrins.h>
#define uchar unsigned char
sbit DQ=P1^0;
bdata uchar dat;
sbit dat0=dat^0;
sbit dat7=dat^7;
uchar dp[16]={0,0,1,1,2,3,3,4,5,5,6,6,7,8,8,9};
uchar code segtab[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e,0xff,0xef,0xbf};
uchar dbuf[4]={0,0,0,0};
void disp(void)
{ uchar i,n,wx;
wx=0xfe;
for(n=0;n<4;n++)
{ P2=wx;
P0=segtab[dbuf[n]];
if(n==1) P0=P0&0x7f;
wx=(wx<<1)+1;
for(i=1;1<200;i++);
P0=0xff;
}
}
void delay(unsigned int i)
{ unsigned int j;
for(j=i;j>0;j--);
}
bit reset(void)
{ bit flag;
DQ=0;
delay(60);
DQ=1;
delay(3);
flag=DQ;
delay(25);
return(flag);
}
void ds18b20_init(void)
{
while(1)
{
if(!reset())
{
DQ=1;
delay(30);
break;
}
else
reset();
}
}
void wrbyte(uchar t)
{
uchar i,dat;
dat=t;
for(i=8;i>0;i--)
{
DQ=0;
_nop_();
DQ=dat&0x01;
dat=dat>>1;
delay(5);
DQ=1;
delay(1);
}
}
uchar rdbyte(void)
{ uchar i;
uchar dat=0;
DQ=1;_nop_();
for(i=8;i>0;i--)
{
DQ=0;
dat>>=1;
DQ=1;_nop_();_nop_();
if(DQ) dat|=0x80;
delay(30);
DQ=1;
_nop_();
}
return(dat);
}
void convert(void)
{
ds18b20_init();
wrbyte(0xcc);
wrbyte(0x44);
delay(40);
}
int readt(void)
{ uchar h,l;
ds18b20_init();
wrbyte(0xcc);
wrbyte(0x44);
l=rdbyte();
h=rdbyte();
return(h*256+l);
}
void tdatacl(int x)
{
bit zf;
zf=0;
if(x<0)
{zf=1;
x=-x;
}
dbuf[0]=dp[x&0x0f];
x=x>>4;
dbuf[3]=x/100;
x=x%100;
dbuf[2]=x/10;
dbuf[1]=x%10;
if(zf==1)
{ if(dbuf[2]==0)
{ dbuf[3]=0xff;
dbuf[2]=0xbf;
}
else
dbuf[3]=0xbf;
}
else
{ if(dbuf[3]==0)
if(dbuf[2]==0)
{dbuf[2]=0xff;
dbuf[3]=0xff;
}
else dbuf[3]=0xff;
}
}
void main()
{
int t;
uchar i;
while(1)
{ convert();
t=rdbyte();
tdatacl(t);
for(i=50;i>0;i--)
disp();
}
}
在仿真时数码管只亮。但是没有数字显示。如果程序不对的话请帮忙改正或者给个正确程序邮箱[email protected]!!!!!在此谢谢了

#include<reg51.h>
#include<intrins.h>
#include <math.H> //要用到取绝对值函数abs()
//通过DS18B20测试当前环境温度, 并通过数码管显示当前温度值, 目前显示范围: -55~ +125度
sbit wela = P2^7; //数码管位选
sbit dula = P2^6; //数码管段选
sbit ds = P2^2;
int tempValue;
//数码管2-5位选
unsigned char code tablewe[]={0xfd,0xfb,0xf7,0xef,0xdf};
//0-F数码管的编码(共阴极)
unsigned char code table[]={
0x3f,0x06,0x5b,0x4f,
0x66,0x6d,0x7d,0x07,
0x7f,0x6f,0x77,0x7c,
0x39,0x5e,0x79,0x71};
//0-9数码管的编码(共阴极), 带小数点
unsigned char code tableWidthDot[]={
0xbf,0x86,0xdb,0xcf,
0xe6,0xed,0xfd,0x87,
0xff,0xef};
//延时函数, 对于11.0592MHz时钟, 例i=10,则大概延时10ms.
void delay(unsigned int i)
{
unsigned int j;
while(i--)
{
for(j = 0; j < 125; j++);
}
}

//初始化DS18B20
//让DS18B20一段相对长时间低电平, 然后一段相对非常短时间高电平, 即可启动
void dsInit()
{
//对于11.0592MHz时钟, unsigned int型的i, 作一个i++操作的时间大于?us
unsigned int i;
ds = 0;
i = 100; //拉低约800us, 符合协议要求的480us以上
while(i>0) i--;
ds = 1; //产生一个上升沿, 进入等待应答状态
i = 4;
while(i>0) i--;
}

void dsWait()
{
unsigned int i;
while(ds);
while(~ds); //检测到应答脉冲
i = 4;
while(i > 0) i--;
}

//向DS18B20读取一位数据
//读一位, 让DS18B20一小周期低电平, 然后两小周期高电平,
//之后DS18B20则会输出持续一段时间的一位数据
bit readBit()
{
unsigned int i;
bit b;
ds = 0;
i++; //延时约8us, 符合协议要求至少保持1us
ds = 1;
i++; i++; //延时约16us, 符合协议要求的至少延时15us以上
b = ds;
i = 8;
while(i>0) i--; //延时约64us, 符合读时隙不低于60us要求
return b;
}

//读取一字节数据, 通过调用readBit()来实现
unsigned char readByte()
{
unsigned int i;
unsigned char j, dat;
dat = 0;
for(i=0; i<8; i++)
{
j = readBit();
//最先读出的是最低位数据
dat = (j << 7) | (dat >> 1);
}
return dat;
}

//向DS18B20写入一字节数据
void writeByte(unsigned char dat)
{
unsigned int i;
unsigned char j;
bit b;
for(j = 0; j < 8; j++)
{
b = dat & 0x01;
dat >>= 1;
//写"1", 将DQ拉低15us后, 在15us~60us内将DQ拉高, 即完成写1
if(b)
{
ds = 0;
i++; i++; //拉低约16us, 符号要求15~60us内
ds = 1;
i = 8; while(i>0) i--; //延时约64us, 符合写时隙不低于60us要求
}
else //写"0", 将DQ拉低60us~120us
ds = 0;
i = 8; while(i>0) i--; //拉低约64us, 符号要求
ds = 1;
i++; i++; //整个写0时隙过程已经超过60us, 这里就不用像写1那样, 再延时64us了

}
}

//向DS18B20发送温度转换命令
void sendChangeCmd()
{
dsInit(); //初始化DS18B20, 无论什么命令, 首先都要发起初始化
dsWait(); //等待DS18B20应答
delay(1); //延时1ms, 因为DS18B20会拉低DQ 60~240us作为应答信号
writeByte(0xcc); //写入跳过序列号命令字 Skip Rom
writeByte(0x44); //写入温度转换命令字 Convert T
}

//向DS18B20发送读取数据命令
void sendReadCmd()
{
dsInit();
dsWait();
delay(1);
writeByte(0xcc); //写入跳过序列号命令字 Skip Rom
writeByte(0xbe); //写入读取数据令字 Read Scratchpad
}

//获取当前温度值
int getTmpValue()
{
unsigned int tmpvalue;
int value; //存放温度数值
float t;
unsigned char low, high;
sendReadCmd();
//连续读取两个字节数据
low = readByte();
high = readByte();
//将高低两个字节合成一个整形变量
//计算机中对于负数是利用补码来表示的
//若是负值, 读取出来的数值是用补码表示的, 可直接赋值给int型的
tmpvalue = high;
tmpvalue <<= 8;
tmpvalue |= low;
value = tmpvalue;

//使用DS18B20的默认分辨率12位, 精确度为0.0625度, 即读回数据的最低位代表0.0625度
t = value * 0.0625;
//将它放大100倍, 使显示时可显示小数点后两位, 并对小数点后第三进行4舍5入
//如t=11.0625, 进行计数后, 得到value = 1106, 即11.06 度
//如t=-11.0625, 进行计数后, 得到value = -1106, 即-11.06 度
value = t * 100 + (value > 0 ? 0.5 : -0.5); //大于0加0.5, 小于0减0.5
return value;
}

unsigned char const timeCount = 3; //动态扫描的时间间隔
//显示当前温度值, 精确到小数点后一位
//若先位选再段选, 由于IO口默认输出高电平, 所以当先位选会使数码管出现乱码
void display(int v)
{
unsigned char count;
unsigned char datas[] = {0, 0, 0, 0, 0};
unsigned int tmp = abs(v);
datas[0] = tmp / 10000;
datas[1] = tmp % 10000 / 1000;
datas[2] = tmp % 1000 / 100;
datas[3] = tmp % 100 / 10;
datas[4] = tmp % 10;
if(v < 0)
{
//关位选, 去除对上一位的影响
wela = 1; //打开锁存, 给它一个下降沿量
P0 = 0;
wela = 0;
//段选
dula = 1; //打开锁存, 给它一个下降沿量
P0 = 0x40; //显示"-"号
dula = 0;

//位选
P0 = 0xfe;//选通第一个数码管
wela = 1; //打开锁存, 给它一个下降沿量
wela = 0;
delay(timeCount);
}
for(count = 0; count != 5; count++)
{
//关位选, 去除对上一位的影响
P0 = 0;
wela = 1; //打开锁存, 给它一个下降沿量
wela = 0;
//段选
if(count != 2)
{

P0 = table[datas[count]]; //显示数字
}
else
{
P0 = tableWidthDot[datas[count]]; //显示带小数点数字
}
dula = 1; //打开锁存, 给它一个下降沿量
dula = 0;

//位选
P0 =tablewe[count]; //选择第(count + 1) 个数码管
wela = 1; //打开锁存, 给它一个下降沿量
wela = 0;
delay(timeCount);
}
}

void main()
{
unsigned char i;

while(1)
{
//启动温度转换
sendChangeCmd();
//显示5次
for(i = 0; i < 40; i++)
{
display(tempValue);
}
tempValue = getTmpValue();
}
}
温馨提示:答案为网友推荐,仅供参考
第1个回答  2012-05-24
#include <ht48f30e.h>

#define uint8 unsigned int //8 bit
#define uint unsigned int //8 bit
#define uint16 unsigned long //16 bit

#define date _pb0//io 0 int 1
#define dateio _15_0//pbc0
#define Dispdata _pb7//io 0 int 1
#define Dispclk _pb6//io 0 int 1

uint8 const showtable[29]={0x7e,0x06,0x6d,0x79,0x33,
0x5b,0xdf,0x70,0x7f,0x7b,0x77,0x1f,0x4e,0x3d,0x4f,0x47,
0x00,0x31,0x07,0x01,0x37,0x0e,0x67,0x63,0x1d,0x15,0x3e,0x4f,0x76};

uint8 const dotcodeh[]={0,0,1,1,2,3,3,4,5,5,6,6,7,8,8,9};

uint tplsb,tpmsb; // 温度值低位、高位字节
uint16 tplmsb;
uint16 temp @0x50;

void delay1ms(uint16 j)
{
uint8 i;
while(j)
{
_clrwdt();
j--;
for(i=0;i<168;i++)
{;}
}
}

void disp(uint8 t,uint8 showdata)
{
unsigned char i,j,temp;

for(j=0;j<t;j++)
{
temp=showdata;
for(i=0; i<7; i++)
{
if(temp & 0x01)
{
Dispdata =1; //数据为1
}
else
{
Dispdata= 0; //数据为0
}
temp>>= 1;
Dispclk=0;
_delay(5);
Dispclk=1;
_delay(5);

}
}
}

void init()
{
uint i;
date=1;
delay1ms(1);

date = 0;
i = 100; //拉低约900us
while (i>0) i--;

date= 1; // 产生上升沿
i = 4;
while (i>0) i--;
}

void wait()
{
uint i;
dateio=1; //改PB5为输入脚

while(date);
while(~date); // 检测到应答脉冲

dateio=0;//改PB5为输出脚

i = 4;
while (i>0) i--;
}

bit readbit()
{
uint i;
bit b;
date = 0;

_nop();
_nop();

date = 1;
dateio=1; //改PB5为输入脚
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
// 延时15us以上,读时隙下降沿后15us,DS18B20输出数据才有效
b = date;
dateio=0;//改PB5为输出脚
i = 8;
while(i>0) i--;
return (b);
}

uint readbyte()
{
uint i,j,b;
b = 0;
for (i=1;i<=8;i++)
{
j = readbit();
b = (j<<7)|(b>>1);//读出的数据最低位在最前面,这样刚好一个字节在DATE里
}
return(b);
}

void writebyte(uint b)
{
uint i;
uint j;
bit btmp;
for(j=1;j<=8;j++)
{
btmp = b&0x01;
b = b>>1; // 取下一位(由低位向高位)
if (btmp) // 写1
{

date = 0;

_nop();// 延时,使得15us以内拉高
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
date = 1;
i = 8;
while(i>0) i--; // 整个写1时隙不低于60us
}
else // 写0
{

date = 0;
i = 8;
while(i>0) i--; // 保持低在60us到120us之间
date = 1;

_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
_nop();
}
}
}

void main()
{
uint8 a1,a2,a3,a4=16;
uint8 m=60,n;
_intc=0;
_tmrc=0;
_wdts=0b00000101 ; //MOV A,00000101B ;512ms
_clrwdt(); //MOV WDTS,A

_pbc=0x00;
delay1ms(1); // 延时1ms

init(); // 产生复位脉冲,初始化DS18B20
wait(); // 等待DS18B20给出应答脉冲
delay1ms(1); // 延时
writebyte(0xcc); // 跳过rom 命令
writebyte(0x4e);//写暂存器
writebyte(0x32);//上限50°C
writebyte(0x00);//下限50°C
writebyte(0x7f);//设置分辨率为12位

//writebyte(0x44); // convert T 命令
delay1ms(1000); // 延时1s

while(1)
{

init(); // 产生复位脉冲,初始化DS18B20
wait(); // 等待DS18B20给出应答脉冲
delay1ms(1); // 延时
writebyte(0xcc); // 跳过rom 命令
writebyte(0x44); // convert T 命令
delay1ms(1000);

init(); // 产生复位脉冲,初始化DS18B20
wait(); // 等待DS18B20给出应答脉冲
delay1ms(1); // 延时
writebyte(0xcc); // 跳过rom命令
writebyte(0xbe); // read scratchpad 读暂存器命令
tplsb=readbyte(); // 温度值低位字节(其中低4位为二进制的"小数"部分)
tpmsb=readbyte(); // 温度值高位字节(其中高5位为符号位)

if(tpmsb>=0xf8)
{ a4=19;
tplsb=~tplsb;
if(tplsb==0xff) {tplsb=0;tpmsb=~tpmsb;++tpmsb;}
else {tplsb=tplsb+1;tpmsb=~tpmsb;}
}

temp=tpmsb;
temp=temp<<8; //合成一个字
temp=temp|tplsb;
a3=temp&0x000f;
temp=temp>>4;
temp=temp&0x00ff;
a1=temp/10;
a2=temp%10;

if(a4==19) disp(1,showtable[a4]);
disp(1,showtable[a1]);
disp(1,showtable[a2]);
disp(1,showtable[dotcodeh[a3]]);
if(a4==16) disp(1,0);
m=60;
n=4;
}
}

、、、、、、、、、、、、、、、、、、、、、、、、、、、
给你参考而已,自己工作使用过的,子程序一定没有问题本回答被提问者和网友采纳
相似回答