【51单片机】程序实验15.DS18B20温度传感器

主要参考学习资料:B站【普中官方】51单片机手把手教学视频

开发资料下载链接:http://www.prechin.cn/gongsixinwen/208.html

单片机套装:普中STC51单片机开发板A4标准版套餐7

DS18B20介绍

主要特性

DS18B20是达拉斯半导体公司推出的单总线接口的传感器,与传统的热敏电阻测温元件相比,它是一种新型的体积小、工作范围宽、与微处理器接口简单的数字化温度传感器,具体特点如下:

①工作电压范围为3.0V~5.5V,除了依靠电源管脚供电,也可采用寄生电源方式(DQ数据线管脚供电)。

②采用单线接口方式,与微处理器只需要一根线即可实现数据通信。

③支持多点组网功能,多个DS18B20可以同时并联在单总线上。

④不需要外围元器件,所有传感器和转换电路都集成在DS18B20中。

⑤工作温度范围为-55℃~125℃,在-10℃~85℃的精度为±0.5℃。

⑥分辨率(测量数据的细微程度)设置范围为912bit,即0.50.0625,分辨率越高,最大转换时间越慢。DS18B20默认分辨率为12bit,最大转换时间750ms。

⑦测量结果直接以数字通过DQ串行输出给CPU,同时可传输CRC校验码,具有较强的抗干扰能力。

⑧具有负压特性,GND和VDD反接时DS18B20不会因发热而烧毁,但不能正常工作。

内部结构

ROM:刻有64位序列号,可以看作是地址序列号,含有8位产品类型标号(28H)、48位DS18B20自身序列号和8位针对前56位的CRC校验码。每个DS18B20的序列号都不相同,以便实现单总线的多点组网。

高速缓存存储器:包括一个高速暂存器RAM和一个EEPROM,后者存放高/低温触发器TH、TL和结构寄存器。

高速暂存存储器:由9个字节组成:

寄存器内容字节地址
温度值低位LS Byte0
温度值高位MS Byte1
高温限值TH2
低温限值TL3
配置寄存器4
保留5
保留6
保留7
CRC校验值8

当温度转换命令44H发布后,经转换所得温度值以二进制补码形式存放在存储器MS和LS中,MS的高5位为符号位,若温度非负,则符号位均为0,将测得的数值乘以分辨率即为实际温度;温度为负时,符号位均为1,测得的数值需先取反后加一再乘以分辨率。

配置寄存器:高3位依次为TM、R1、R0,低5位均为1。TM为测试模式位,用于设置DS18B20在工作模式还是测试模式,在出厂时已被设置为1(工作模式),不宜改动。R1和R0用于设置分辨率,默认均为1,如下表所示:

R1R0精度最大转换时间
009bit93.75ms
0110bit187.5ms
1011bit375ms
1112bit750ms

控制时序

单总线器件需要严格的信号时序来保证数据的完整性,DS18B20的时序包括初始化时序、写时序、读时序。DS18B20发送命令和数据的字节均是低位在前。

初始化时序

单总线的所有通信均以初始化时序开始。①主机将总线先置高后拉低,保持480960μs。②主机释放总线,电阻上拉,等待DS18B20在1560μs后响应。③DS18B20拉低总线,保持60~240μs。④主机读取到低电平后继续延时,延时时间从释放总线算起至少480μs。

写时序

写时序包括写0和写1,所有写时序的过程至少需要60μs,且两次独立的写时序之间至少需要1μs的恢复时间。写时序均起始于主机拉低总线,若写0则延时60μs后拉高总线并继续延时2μs;若写1则延时大概2μs后拉高总线并继续延时60μs。

读时序

单总线器件仅在主机发出读时序后才向主机发送数据,所以在主机发出读取命令之后必须马上产生读时序。读时序的过程也至少需要60μs,两次独立的读时序之间至少需要1μs的恢复时间。读时序均由主机产生至少1μs的低电平发起,随后释放总线,并在其后的15μs之间采集总线状态,低电平则读0,高电平则读1。

DS18B20典型的温度读取过程:复位(初始化时序)→发动SKIP ROM命令(0XCC)→发送开始转换命令(0X44)→延时→复位→发送SKIP ROM命令(0XCC)→发送读存储器命令(0XBE)→连续读出两个字节的温度数据→结束。

硬件设计

本实验板载DS18B20传感器的数据总线绑定到P3.7IO口管脚。

实验15 DS18B20温度传感器

实现功能:插上DS18B20温度传感器,数码管显示检测的温度值。

DS18B20驱动

按照实验14中的多文件工程框架,在App>ds18b20中创建:

ds18b20.h

#ifndef _DS18B20_H
#define _DS18B20_H

#include "public.h"

//定义IO管脚
sbit DS18B20_PORT = P3^7;

//声明主函数需要的源文件函数
u8 ds18b20_init(void);
float ds18b20_read_temperature(void);

#endif

ds18b20.c

#include "ds18b20.h"
//通过intrins.h调用延时2μs需要用到的nop函数
#include "intrins.h"

//复位函数
void ds18b20_reset(void)
{
	DS18B20_PORT = 0;
	delay_10us(75);
	DS18B20_PORT = 1;
	delay_10us(2);
}

/*
检测DS18B20函数是否存在
若一直为高电平或低电平则不存在,返回1,否则返回0
*/
u8 ds18b20_check(void)
{
	//计时器变量
	u8 time_temp = 0;
	//检测低电平
	while(DS18B20_PORT && time_temp<20)
	{
		time_temp++;
		delay_10us(1);
	}
	//若等待超时则返回1,否则重置计时器
	if(time_temp >= 20)return 1;
	else time_temp = 0;
	//检测高电平
	while((!DS18B20_PORT) && (time_temp<20))
	{
		time_temp++;
		delay_10us(1);
	}
	//若等待超时则返回1,否则返回0
	if(time_temp >= 20)return 1;
	return 0;
}

//初始化同时检测存在
u8 ds18b20_init(void)
{
	ds18b20_reset();
	return ds18b20_check();
}

//写字节函数
void ds18b20_write_byte(u8 dat)
{
	//8位输入循环控制变量
	u8 i = 0;
	//存储准备输入的位的变量
	u8 temp = 0;
	//从低到高按位输入
	for(i=0;i<8;i++)
	{
		//和00000001进行与运算提取最低位
		temp = dat & 0X01;
		//字节右移将次低位移到最低位
		dat >>= 1;
		//根据temp写1或写0
		if(temp)
		{
			//拉低总线
			DS18B20_PORT = 0;
			//延时2μs,nop函数延时1个机器周期,12MHz下即为1μs
			_nop_();_nop_();
			//写入1
			DS18B20_PORT = 1;
			//延时60μs
			delay_10us(6);
		}
		else
		{
			//拉低总线写入0
			DS18B20_PORT = 0;
			//延时60μs
			delay_10us(6);
			//释放总线
			DS18B20_PORT = 1;
			//延时2μs
			_nop_();_nop_();
		}
	}
}

//读取一个位
u8 ds18b20_read_bit(void)
{
	//存储读取数据的变量
	u8 dat = 0;
	//拉低总线
	DS18B20_PORT = 0;
	//延时2μs
	_nop_();_nop_();
	//释放总线后读取
	DS18B20_PORT = 1;
	if(DS18B20_PORT)dat = 1;
	else dat = 0;
	delay_10us(5);
	return dat;
}

//读取一字节
u8 ds18b20_read_byte(void)
{
	//8位读取循环控制变量
	u8 i = 0;
	//存储读取到的每一位的变量
	u8 temp = 0;
	//将每一位存储进字节的变量
	u8 dat = 0;
	//从低到高按位读取
	for(i=0;i<8;i++)
	{
		//读取一位
		temp = ds18b20_read_bit();
		//字节右移一位并将新位存储到最高位
		dat >>= 1;
		//或运算防止其他位被擦除
		dat |= temp << 7;
	}
	return dat;
}

//温度读取函数
float ds18b20_read_temperature(void)
{
	//存储高八位数值
	u8 dath = 0;
	//存储低八位数值
	u8 datl = 0;
	//存储完整十六位数值
	u16 value = 0;
	//存储转换得到的实际温度
	float temp = 0;
	//参照读时序介绍中的温度读取过程
	ds18b20_reset();
	ds18b20_check();
	ds18b20_write_byte(0XCC);
	ds18b20_write_byte(0X44);
	ds18b20_reset();
	ds18b20_check();
	ds18b20_write_byte(0XCC);
	ds18b20_write_byte(0XBE);
	//先读低八位,后读高八位
	datl = ds18b20_read_byte();
	dath = ds18b20_read_byte();
	//高八位移到十六位中的高八位再与低八位结合得到完整的十六位数值
	value = (dath << 8) + datl;
	//和1111100000000000与运算检测符号位
	if((value&&0XF800) == 0XF800)
	{
		//符号位均为1则取反加一得到正值
		value = (~value) + 1;
		//计算实际温度
		temp = value * (-0.0625);
	}
	else temp = value * 0.0625;
	return temp;
}

数码管驱动

数码管驱动延用实验14,但为了数码管显示更加自定义,可以显示负号、小数点等符号,段码值转换功能将移到函数外部来完成,即参数dat直接传入段码值,以下是smg.c修改后的程序:

#include "smg.h"

u8 gseg_code[16] = {0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x07, 0x7f, 0x6f, 0x77, 0x7c, 0x39, 0x5e, 0x79, 0x71};

void seg_display(u8 dat[], u8 pos)
{
	u8 i = 0;
	u8 pos_temp = pos - 1;
	for(i = pos_temp;i < 8;i++)
	{
		switch(i)
		{
			case 0: LSC = 1;LSB = 1;LSA = 1;break;
			case 1: LSC = 1;LSB = 1;LSA = 0;break;
			case 2: LSC = 1;LSB = 0;LSA = 1;break;
			case 3: LSC = 1;LSB = 0;LSA = 0;break;
			case 4: LSC = 0;LSB = 1;LSA = 1;break;
			case 5: LSC = 0;LSB = 1;LSA = 0;break;
			case 6: LSC = 0;LSB = 0;LSA = 1;break;
			case 7: LSC = 0;LSB = 0;LSA = 0;break;
		}
		//此处删去gesg_code[]
		SEG_A_DP_PORT = dat[i-pos_temp];
		delay_10us(100);
		SEG_A_DP_PORT = 0x00;
	}
}

主函数

#include "public.h"
#include "smg.h"
#include "ds18b20.h"

void main()
{
	//读取温度间隔的延时控制变量
	u8 i = 0;
	//本实验对温度保留一位小数处理,方法为先乘十后提取整数部分,在数码管上添加小数点显示
	int temp_value = 0;
	//存储每一位供数码管显示的段码值,温度测量结果最高占用5位数码管
	u8 temp_buf[5];
	//初始化
	ds18b20_init();
	while(1)
	{
		//计时器累加
		i++;
		//计时器延时50ms后读取,可自行调整
		if(i%50 == 0)temp_value = ds18b20_read_temperature() * 10;
		//判断温度正负
		if(temp_value < 0)
		{
			//若温度为负,将温度转为正以便拆解数位,并使第一个数码管显示负号
			temp_value = -temp_value;
			temp_buf[0] = 0X40;
		}
		else
			//若温度为正,第一个数码管不显示
			temp_buf[0] = 0X00;
		//将温度拆解数位并转换为段码值依次存储
		temp_buf[1] = gseg_code[temp_value / 1000];
		temp_buf[2] = gseg_code[temp_value % 1000 / 100];
		temp_buf[3] = gseg_code[temp_value % 100 / 10] | 0X80;
		temp_buf[4] = gseg_code[temp_value % 10];
		//将温度显示在数码管后5位上
		seg_display(temp_buf, 4);
	}
}
<think>好的,我现在需要处理用户关于DS18B20温度传感器在C51单片机上的使用方法和代码实现的查询。首先,我要确认用户的需求是什么。他们可能想要连接DS18B20到C51单片机,并获取温度数据,同时需要示例代码和教程。根据用户提供的引用资料,引用[1]提到了连接方式,引用[3]中有代码片段,而引用[2]则介绍了DS18B20的特点。 接下来,我应该组织信息,分步骤回答。首先需要说明硬件连接,这是基础。引用[1]提到VCC、GND和DQ引脚的连接,所以这部分可以直接引用。然后,需要讲解驱动程序的结构,比如初始化、复位、读写操作等。引用[3]中的代码展示了如何获取温度,但可能需要更详细的解释。用户可能需要完整的代码示例,所以需要整合引用中的代码,并补充必要的部分,比如延时函数和主程序。 另外,用户可能对温度转换和数据处理有疑问,比如如何将读取的二进制数据转换为实际温度值。引用[3]中的代码处理了温度值的转换,包括补码处理和精度调整,这部分需要详细解释,让用户明白每一步的作用。 还需要注意代码中的关键点,比如跳过ROM命令(0xCC)、启动温度转换(0x44)、读取温度(0xBE)等,这些是DS18B20的标准操作流程。同时,时序控制非常重要,因为DS18B20对时序要求严格,代码中的延时函数必须准确,否则可能导致读取失败。 可能用户在实际操作中会遇到问题,比如传感器无响应或数据错误,因此在回答中可以加入常见问题及解决方法,比如检查接线是否正确,确保延时函数的准确性,或者是否需要上拉电阻等。 最后,生成相关问题时,要考虑到用户可能进一步想了解如何优化代码、如何显示温度数据或者如何应用在具体项目中,比如PID控制系统,如引用[3]提到的。因此,相关问题可以涉及这些方面,帮助用户深入学习和解决问题。</think>以下是根据DS18B20温度传感器在C51单片机应用的完整说明: # DS18B20温度传感器C51驱动程序实现 ## 一、硬件连接说明 根据引用[1]的接线方式: 1. 连接DS18B20的VCC到单片机5V电源 2. GND连接共地 3. DQ数据线接P1.0(可自定义IO口) 4. 建议在DQ线增加4.7K上拉电阻 ![接线示意图](https://via.placeholder.com/300x200/说明图暂缺) ## 二、驱动程序核心代码 ```c #include <REG52.H> sbit DQ = P1^0; // 定义通信引脚 // 延时函数(12MHz晶振) void delay_us(unsigned int us) { while(us--); } // DS18B20复位 bit ds18b20_rst() { bit ack; DQ = 0; delay_us(480); // 480μs低电平复位信号 DQ = 1; delay_us(60); // 释放总线等待60μs ack = DQ; // 检测应答信号 delay_us(420); // 等待复位周期完成 return ~ack; // 返回0表示存在 } // 写入1字节 void ds18b20_write(unsigned char dat) { for(int i=0; i<8; i++){ DQ = 0; DQ = dat & 0x01; // 从低位开始传输 delay_us(60); // 保持60μs DQ = 1; dat >>= 1; delay_us(10); // 恢复时间 } } // 读取1字节 unsigned char ds18b20_read() { unsigned char dat = 0; for(int i=0; i<8; i++){ DQ = 0; delay_us(2); // 保持2μs DQ = 1; // 释放总线 delay_us(8); // 等待8μs后采样 dat |= (DQ << i); // 读取数据位 delay_us(50); // 完成时隙 } return dat; } // 获取温度值(带符号整数) int get_temperature() { unsigned char LSB, MSB; int temp; ds18b20_rst(); ds18b20_write(0xCC); // 跳过ROM[^3] ds18b20_write(0x44); // 启动温度转换 delay_us(200); // 等待转换完成 ds18b20_rst(); ds18b20_write(0xCC); ds18b20_write(0xBE); // 读取暂存器 LSB = ds18b20_read(); MSB = ds18b20_read(); temp = MSB << 8 | LSB; if(temp & 0x8000) { // 处理负温度 temp = ~temp + 1; temp = -temp; } return temp * 0.0625; // 转换为实际温度值 } ``` ## 三、关键参数说明 1. 温度分辨率:可通过配置寄存器设置为9-12位(默认12位) 2. 转换时间:12位分辨率时约750ms 3. 温度范围:$-55^{\circ}C$ 到 $+125^{\circ}C$,精度$\pm0.5^{\circ}C$[^2] ## 四、典型应用场景 ```c void main() { int temperature; while(1) { temperature = get_temperature(); // LCD显示或串口输出温度值 delay_ms(1000); // 每秒更新 } } ``` ## 五、常见问题解决 1. **无响应**:检查接线、上拉电阻、延时函数精度 2. **数据错误**:确保时序符合DS18B20规范 3. **负温度异常**:注意补码转换处理
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值