c语言void ds,单片机中使用DS18B20温度传感器C语言程序(参考7)

硬件: 51板

(1)单线ds18b20接 P2.2

(2)使用外部电源给ds18b20供电,没有使用寄生电源奥

软件:

Kei uVision 2

#include "reg52.h"

#include "intrins.h"

#define uchar unsigned char

#define uint unsigned int

sbit ds=P2^2;

sbit dula=P2^6;

sbit wela=P2^7;

uchar flag ;

uint temp;                //参数temp一定要声明为 int 型

uchar code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,

0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};      //不带小数点数字编码

uchar code table1[]={0xbf,0x86,0xdb,0xcf,0xe6,0xed,0xfd,

0x87,0xff,0xef};         //带小数点数字编码

/*延时函数*/

void TempDelay (uchar us)

{

while(us--);

}

void delay(uint count) //延时子函数

{

uint i;

while(count)

{

i=200;

while(i>0)

i--;

count--;

}

}

/*串口初始化,波特率9600,方式1 */

void init_com()

{

TMOD=0x20;       //设置定时器1为模式2

TH1=0xfd;        //装初值设定波特率

TL1=0xfd;

TR1=1;           //启动定时器

SM0=0;           //串口通信模式设置

SM1=1;

// REN=1;           //串口允许接收数据

PCON=0;          //波特率不倍频

//   SMOD=0;        //波特率不倍频

// EA=1;           //开总中断

//ES=1;           //开串行中断

}

/*数码管的显示 */

void display(uint temp)

{

uchar bai,shi,ge;

bai=temp/100;

shi=temp%100/10;

ge=temp%100%10;

dula=0;

P0=table[bai];   //显示百位

dula=1;          //从0到1,有个上升沿,解除锁存,显示相应段

dula=0;          //从1到0再次锁存

wela=0;

P0=0xfe;

wela=1;

wela=0;

delay(1);     //延时约2ms

P0=table1[shi];   //显示十位

dula=1;

dula=0;

P0=0xfd;

wela=1;

wela=0;

delay(1);

P0=table[ge];   //显示个位

dula=1;

dula=0;

P0=0xfb;

wela=1;

wela=0;

delay(1);

}

/*****************************************

时序:初始化时序、读时序、写时序。

所有时序都是将主机(单片机)作为主设备,单总

线器件作为从设备。而每一次命令和数据的传输

都是从主机主动启动写时序开始,如果要求单总

线器件回送数据,在进行写命令后,主机需启动

读时序完成数据接收。数据和命令的传输都是低

位在先。

初始化时序:复位脉冲 存在脉冲

读;1 或 0时序

写;1 或 0时序

只有存在脉冲信号是从18b20(从机)发出的,其

它信号都是由主机发出的。

存在脉冲:让主机(总线)知道从机(18b20)已

经做好了准备。

******************************************/

/*--------------------------------------------------------------------------------------------------------------------

初始化:检测总线控制器发出的复位脉冲

和ds18b20的任何通讯都要从初始化开始

初始化序列包括一个由总线控制器发出的复位脉冲

和跟在其后由从机发出的存在脉冲。

初始化:复位脉冲+存在脉冲

具体操作:

总线控制器发出(TX)一个复位脉冲 (一个最少保持480μs 的低电平信号),然后释放总线,

进入接收状态(RX)。单线总线由5K 上拉电阻拉到高电平。探测到I/O 引脚上的上升沿后

DS1820 等待15~60μs,然后发出存在脉冲(一个60~240μs 的低电平信号)。

具体看"倒塌 18b20"文档里的 " 单线复位脉冲时序和1-wire presence detect "的时序图

-------------------------------------------------------------------------------------------------------------------*/

void ds_reset(void)

{

ds=1;

_nop_();         //1us

ds=0;

TempDelay(80);   //当总线停留在低电平超过480us,总线上所以器件都将被复位,这里//延时约530us总线停留在低电平超过480μs,总线上的所有器件都

//将被复位。

_nop_();

ds=1;            //产生复位脉冲后,微处理器释放总线,让总线处于空闲状态,原因查//18b20中文资料

TempDelay(5);   //释放总线后,以便从机18b20通过拉低总线来指示其是否在线,

//存在检测高电平时间:15~60us, 所以延时44us,进行             1-wire presence //detect(单线存在检测)

_nop_();

_nop_();

_nop_();

if(ds==0)

flag=1;        //detect 18b20 success

else

flag=0;        //detect 18b20 fail

TempDelay(20);     //存在检测低电平时间:60~240us,所以延时约140us

_nop_();

_nop_();

ds=1;           //再次拉高总线,让总线处于空闲状态

/**/

}

/*----------------------------------------

读/写时间隙:

DS1820 的数据读写是通过时间隙处理

位和命令字来确认信息交换。

------------------------------------------*/

bit   ds_read_bit(void)     //读一位

{

bit dat;

ds=0;          //单片机(微处理器)将总线拉低

_nop_();        //读时隙起始于微处理器将总线拉低至少1us

ds=1;         //拉低总线后接着释放总线,让从机18b20能够接管总线,输出有效数据

_nop_();

_nop_();           //小延时一下,读取18b20上的数据 ,因为从ds18b20上输出的数据

//在读"时间隙"下降沿出现15us内有效

dat=ds;            //主机读从机18b20输出的数据,这些数据在读时隙的下降沿出现//15us内有效

TempDelay(10);     //所有读"时间隙"必须60~120us,这里77us

return(dat);        //返回有效数据

}

uchar ds_read_byte(void ) //读一字节

{

uchar value,i,j;

value=0;            //一定别忘了给初值

for(i=0;i<8;i++)

{

j=ds_read_bit();

value=(j<<7)|(value>>1);    //这一步的说明在一个word文档里面

}

return(value);         //返回一个字节的数据

}

void ds_write_byte(uchar dat) //写一个字节

{

uchar i;

bit onebit;         //一定不要忘了,onebit是一位

for(i=1;i<=8;i++)

{

onebit=dat&0x01;

dat=dat>>1;

if(onebit)       //写 1

{

ds=0;

_nop_();

_nop_();       //看时序图,至少延时1us,才产生写"时间隙"

ds=1;        //写时间隙开始后的15μs内允许数据线拉到高电平

TempDelay(5);   //所有写时间隙必须最少持续60us

}

else          //写 0

{

ds=0;

TempDelay(8);     //主机要生成一个写0 时间隙,必须把数据线拉到低电平并保持至少60μs,这里64us

ds=1;

_nop_();

_nop_();

}

}

}

/*****************************************

主机(单片机)控制18B20完成温度转换要经过三个步骤:

每一次读写之前都要18B20进行复位操作,复位成功后发送

一条ROM指令,最后发送RAM指令,这样才能对DS18b20进行

预定的操作。

复位要求主CPU将数据线下拉500us,然后释放,当ds18B20

受到信号后等待16~60us,后发出60~240us的存在低脉冲,

主CPU收到此信号表示复位成功

******************************************/

/*----------------------------------------

进行温度转换:

先初始化

然后跳过ROM:跳过64位ROM地址,直接向ds18B20发温度转换命令,适合单片工作

发送温度转换命令

------------------------------------------*/

void tem_change()

{

ds_reset();

delay(1);               //约2ms

ds_write_byte(0xcc);

ds_write_byte(0x44);

}

/*----------------------------------------

获得温度:

------------------------------------------*/

uint get_temperature()

{

float wendu;

uchar a,b;

ds_reset();

delay(1);               //约2ms

ds_write_byte(0xcc);

ds_write_byte(0xbe);

a=ds_read_byte();

b=ds_read_byte();

temp=b;

temp<<=8;

temp=temp|a;

wendu=temp*0.0625;      //温度读取的解释我记录在 "倒塌 18B20"里面

temp=wendu*10+0.5;

return temp;

}

/*----------------------------------------

读ROM

------------------------------------------*/

/*

void ds_read_rom()                   //这里没有用到

{

uchar a,b;

ds_reset();

delay(30);

ds_write_byte(0x33);

a=ds_read_byte();

b=ds_read_byte();

}

*/

void main()

{

uint a;

init_com();

while(1)

{

tem_change();           //12位转换时间最大为750ms

for(a=10;a>0;a--)

{

display( get_temperature());

}

}

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值