单片机练习 - DS18B20温度转换与显示

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了单片机练习 - DS18B20温度转换与显示相关的知识,希望对你有一定的参考价值。

DS18B20 数字温度传感器(参考:智能温度传感器DS18B20的原理与应用)是DALLAS 公司生产的1-Wire,即单总线器件,具有线路简单,体积小的特点。因此用它来组成一个测温系统,具有线路简单,在一根通信线,可以挂很多这样的数字温度计。DS18B20 产品的特点:

(1)、只要求一个I/O 口即可实现通信。
(2)、在DS18B20 中的每个器件上都有独一无二的序列号。
(3)、实际应用中不需要外部任何元器件即可实现测温。
(4)、测量温度范围在-55 到+125℃之间; 在-10 ~ +85℃范围内误差为±5℃;
(5)、数字温度计的分辨率用户可以从9 位到12 位选择。将12位的温度值转换为数字量所需时间不超过750ms;
(6)、内部有温度上、下限告警设置。

DS18B20引脚分布图
技术分享图片

DS18B20 详细引脚功能描述:
1、GND 地信号;
2、DQ数据输入出引脚。开漏单总线接口引脚。当被用在寄生电源下,此引脚可以向器件提供电源;漏极开路, 常太下高电平. 通常要求外接一个约5kΩ的上拉电阻.
3、VDD可选择的VDD 引脚。电压范围:3~5.5V; 当工作于寄生电源时,此引脚必须接地。

DS18B20存储器结构图
技术分享图片
暂存储器的头两个字节为测得温度信息的低位和高位字节; 
第3, 4字节是TH和TL的易失性拷贝, 在每次电复位时都会被刷新;
第5字节是配置寄存器的易失性拷贝, 同样在电复位时被刷新;
第9字节是前面8个字节的CRC检验值.

配置寄存器的命令内容如下:

0 R1 R0 1 1 1 1 1
MSB                                                                LSB
R0和R1是温度值分辨率位, 按下表进行配置.默认出厂设置是R1R0 = 11, 即12位.

温度值分辨率配置表
R1 R0 分辨率 最大转换时间(ms)
0 0 9bit 93.75(tconv/8)
0 1 10bit 183.50(tconv/4)
1 0 11bit 375(tconv/2)
1 1 12bit 750 (tconv)
4种分辨率对应的温度分辨率为0.5℃, 0.25℃, 0.125℃, 0.0625℃(即最低一位代表的温度值)

12位分辨率时的两个温度字节的具体格式如下:
低字节:
2^3 2^2 2^1 2^0 2^-1 2^-2 2^-3 2^-4

高字节:
S S S S S 2^6 2^5 2^4

其中高字节前5位都是符号位S, 若分辨率低于12位时, 相应地使最低为0, 如: 当分辨率为10位时, 低字节为:
2^3 2^2 2^1 2^0 2^-1 2^-2 0 0

, 高字节不变....

一些温度与转换后输出的数字参照如下:
温度 数字输出 换成16进制
+125℃ 00000111 11010000  07D0H
+85℃ 00000101 01010000 0550H
+25.0625℃ 00000001 10010001 0191H
+10.125℃ 00000000 10100010 00A2H
+0.5℃ 00000000 00001000 0008H
0℃ 00000000 00000000 0000H
-0.5℃ 11111111 11111000 FFF8H
-10.125℃ 11111111 01011110 FFE5H
-25.0625℃ 11111110 01101111 FF6FH
-55℃ 11111100 10010000 FC90H
由上表可看出, 当输出是负温度时, 使用补码表示, 方便计算机运算(若是用C语言, 直接将结果赋值给一个int变量即可).

DS18B20 的使用方法:
由于DS18B20 采用的是1-Wire 总线协议方式,即在一根数据线实现数据的双向传输,而对单片机来说,我们必须采用软件的方法来模拟单总线的协议时序来完成对DS18B20芯片的访问。
由于DS18B20是在一根I/O线上读写数据,因此,对读写的数据位有着严格的时序要求
DS18B20有严格的通信协议来保证各位数据传输的正确性和完整性。
该协议定义了几种信号的时序:初始化时序(dsInit()实现)、读时序(readByte())、写时序(writeByte())
所有时序都是将主机作为主设备,单总线器件作为从设备。而每一次命令和数据的传输都是从主机主动启动写时序开始,如果要求单总线器件回送数据,在进行写命令后,主机需启动读时序完成数据接收。数据和命令的传输都是低位在先。

DS18B20与单片机连接电路图:
技术分享图片

利用软件模拟DS18B20的单线协议和命令:主机操作DS18B20必须遵循下面的顺序
1. 初始化
单线总线上的所有操作都是从初始化开始的. 过程如下: 
1)请求: 主机通过拉低单线480us以上, 产生复位脉冲, 然后释放该线, 进入Rx接收模式. 主机释放总线时, 会产生一个上升沿脉冲.
DQ : 1 -> 0(480us+) -> 1  
2)响应: DS18B20检测到该上升沿后, 延时15~60us, 通过拉低总线60~240us来产生应答脉冲. 
DQ: 1(15~60us) -> 0(60~240us)
3)接收响应: 主机接收到从机的应答脉冲后, 说明有单线器件在线. 至此, 初始化完成.
DQ: 0

2. ROM操作命令
当主机检测到应答脉冲, 便可发起ROM操作命令. 共有5类ROM操作命令, 如下表
 命令类型  命令字节 功能
Read Rom   读ROM   33H 读取激光ROM中的64位,只能用于总线上单个DS18B20器件情况, 多挂时会发生数据冲突
Match Rom匹配ROM 55H 此命令后跟64位ROM序列号,寻址多挂总线上的对应DS18B20.只有序列号完全匹配的DS18B20才能响应后面的内存操作命令,其他不匹配的将等待复位脉冲.可用于单挂或多挂两种情况.
Skip Rom   跳过ROM CCH 可无须提供64位ROM序列号即可运行内存操作命令, 只能用于单挂.
Search Rom搜索ROM F0H 通过一个排除法过程, 识别出总线上所有器件的ROM序列号
Alarm Search告警搜索 ECH 命令流程与Search Rom相同, 但DS18B20只有最近的一次温度测量时满足了告警触发条件的, 才会响应此命令.

3. 内存操作命令
在成功执行ROM操作命令后, 才可使用内存操作命令. 共有6种内存操作命令:
命令类型 命令字节 功能

Write Scratchpad
写暂存器

4EH 写暂存器中地址2~地址4的3个字节(TH,TL和配置寄存器)在发起复位脉冲之前,3个字节都必须要写.

Read Scratchpad
读暂存器

BEH 读取暂存器内容,从字节0~一直到字节8, 共9个字节,主机可随时发起复位脉冲,停止此操作,通常我们只需读前5个字节.

Copy Scratchpad
复制暂存器

48H 将暂存器中的内容复制进EERAM, 以便将温度告警触发字节存入非易失内存. 如果此命令后主机产生读时隙, 那么只要器件还在进行复制都会输出0, 复制完成后输出1.

Convert T 
温度转换

44H 开始温度转换操作. 若在此命令后主机产生时隙, 那么只要器件还在进行温度转换就会输出0, 转换完成后输出1.

Recall E2    
重调E2暂存器

B8H 将存储在EERAM中的温度告警触发值和配置寄存器值重新拷贝到暂存器中,此操作在DS18B20加电时自动产生.

Read Power Supply
读供电方式

B4H 主机发起此命令后每个读数时隙内,DS18B20会发信号通知它的供电方式:0寄生电源, 1外部供电.

4. 数据处理
DS18B20要求有严格的时序来保证数据的完整性. 在单线DQ上, 有复位脉冲, 应答脉冲, 写0, 写1, 读0, 读1这6种信号类型. 除了应答脉冲外, 其它都由主机产生. 数据位的读和写是通过读、写时隙实现的.
1) 写时隙: 当主机将数据线从高电平拉至低电平时, 产生写时隙.所有写时隙都必须在60us以上, 各写时隙间必须保证1us的恢复时间.
写"1" : 主机将数据线DQ先拉低, 然后释放15us后, 将数据线DQ拉高;
写"0" : 主机将DQ拉低并至少保持60us以上.
2)读时隙: 当主机将数据线DQ从高电平拉至低电平时, 产生读时隙. 所有读时隙最短必须持续60us, 各读时隙间必须保证1us的恢复时间.
读: 主机将DQ拉低至少1us,. 此时主机马上将DQ拉高, 然后就可以延时15us后, 读取DQ即可.


源代码: (测量范围: 0 ~ 99度)

 

技术分享图片DS18B20
  1技术分享图片#include <reg51.H>
  2技术分享图片//通过DS18B20测试当前环境温度, 并通过数码管显示当前温度值
  3技术分享图片sbit wela = P2^7;  //数码管位选
  4技术分享图片sbit dula = P2^6;  //数码管段选
  5技术分享图片sbit ds = P2^2;
  6技术分享图片//0-F数码管的编码(共阴极)
  7技术分享图片unsigned char code table[]={0x3f,0x06,0x5b,0x4f,0x66,
  8技术分享图片    0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
  9技术分享图片//0-9数码管的编码(共阴极), 带小数点
 10技术分享图片unsigned char code tableWidthDot[]={0xbf, 0x86, 0xdb, 0xcf, 0xe6, 0xed, 0xfd, 
 11技术分享图片    0x87, 0xff, 0xef};
 12技术分享图片
 13技术分享图片//延时函数, 例i=10,则大概延时10ms.
 14技术分享图片void delay(unsigned char i)
 15技术分享图片{
 16技术分享图片    unsigned char j, k;
 17技术分享图片    for(j = i; j > 0; j--)
 18技术分享图片    {
 19技术分享图片        for(k = 125; k > 0; k--);
 20技术分享图片    }
 21技术分享图片}
 22技术分享图片
 23技术分享图片//初始化DS18B20
 24技术分享图片//让DS18B20一段相对长时间低电平, 然后一段相对非常短时间高电平, 即可启动
 25技术分享图片void dsInit()
 26技术分享图片{
 27技术分享图片    //一定要使用unsigned int型, 一个i++指令的时间, 作为与DS18B20通信的小时间间隔
 28技术分享图片    //以下都是一样使用unsigned int型
 29技术分享图片    unsigned int i;  
 30技术分享图片    ds = 0;
 31技术分享图片    i = 103;
 32技术分享图片    while(i>0) i--;
 33技术分享图片    ds = 1;
 34技术分享图片    i = 4;
 35技术分享图片    while(i>0) i--;
 36技术分享图片}
 37技术分享图片
 38技术分享图片//向DS18B20读取一位数据
 39技术分享图片//读一位, 让DS18B20一小周期低电平, 然后两小周期高电平, 
 40技术分享图片//之后DS18B20则会输出持续一段时间的一位数据
 41技术分享图片bit readBit()
 42技术分享图片{
 43技术分享图片    unsigned int i;
 44技术分享图片    bit b;
 45技术分享图片    ds = 0;
 46技术分享图片    i++;
 47技术分享图片    ds = 1; 
 48技术分享图片    i++; i++;
 49技术分享图片    b = ds;
 50技术分享图片    i = 8; 
 51技术分享图片    while(i>0) i--;
 52技术分享图片    return b;
 53技术分享图片}
 54技术分享图片
 55技术分享图片//读取一字节数据, 通过调用readBit()来实现
 56技术分享图片unsigned char readByte()
 57技术分享图片{
 58技术分享图片    unsigned int i;
 59技术分享图片    unsigned char j, dat;
 60技术分享图片    dat = 0;
 61技术分享图片    for(i=0; i<8; i++)
 62技术分享图片    {
 63技术分享图片        j = readBit();
 64技术分享图片        //最先读出的是最低位数据
 65技术分享图片        dat = (j << 7) | (dat >> 1);
 66技术分享图片    }
 67技术分享图片    return dat;
 68技术分享图片}
 69技术分享图片
 70技术分享图片//向DS18B20写入一字节数据
 71技术分享图片void writeByte(unsigned char dat)
 72技术分享图片{
 73技术分享图片    unsigned int i;
 74技术分享图片    unsigned char j;
 75技术分享图片    bit b;
 76技术分享图片    for(j = 0; j < 8; j++)
 77技术分享图片    {
 78技术分享图片        b = dat & 0x01;
 79技术分享图片        dat >>= 1;
 80技术分享图片        //写"1", 让低电平持续2个小延时, 高电平持续8个小延时
 81技术分享图片        if(b)   
 82技术分享图片        {
 83技术分享图片            ds = 0;
 84技术分享图片            i++; i++;
 85技术分享图片            ds = 1;
 86技术分享图片            i = 8; while(i>0) i--;
 87技术分享图片        }
 88技术分享图片        else  //写"0", 让低电平持续8个小延时, 高电平持续2个小延时
 89技术分享图片        {
 90技术分享图片            ds = 0;
 91技术分享图片            i = 8; while(i>0) i--;
 92技术分享图片            ds = 1;
 93技术分享图片            i++; i++;
 94技术分享图片        }
 95技术分享图片    }
 96技术分享图片}
 97技术分享图片
 98技术分享图片//向DS18B20发送温度转换命令
 99技术分享图片void sendChangeCmd()
100技术分享图片{
101技术分享图片    dsInit();    //初始化DS18B20
102技术分享图片    delay(1);    //延时1ms
103技术分享图片    writeByte(0xcc); //写入跳过序列号命令字
104技术分享图片    writeByte(0x44); //写入温度转换命令字
105技术分享图片}
106技术分享图片
107技术分享图片//向DS18B20发送读取数据命令
108技术分享图片void sendReadCmd()
109技术分享图片{
110技术分享图片    dsInit();
111技术分享图片    delay(1);
112技术分享图片    writeByte(0xcc); //写入跳过序列号命令字
113技术分享图片    writeByte(0xbe); //写入读取数据令字
114技术分享图片}
115技术分享图片
116技术分享图片//获取当前温度值
117技术分享图片unsigned int getTmpValue()
118技术分享图片{
119技术分享图片    unsigned int value; //存放温度数值
120技术分享图片    float t;
121技术分享图片    unsigned char low, high;
122技术分享图片    sendReadCmd();
123技术分享图片    //连续读取两个字节数据
124技术分享图片    low = readByte(); 
125技术分享图片    high = readByte();
126技术分享图片    //将高低两个字节合成一个整形变量
127技术分享图片    value = high;
128技术分享图片    value <<= 8;
129技术分享图片    value |= low;
130技术分享图片    //DS18B20的精确度为0.0625度, 即读回数据的最低位代表0.0625度
131技术分享图片    t = value * 0.0625;
132技术分享图片    //将它放大10倍, 使显示时可显示小数点后一位, 并对小数点后第二2进行4舍5入
133技术分享图片    //如t=11.0625, 进行计数后, 得到value = 111, 即11.1 度
134技术分享图片    value = t * 10 + 0.5;
135技术分享图片    return value;
136技术分享图片}
137技术分享图片
138技术分享图片//显示当前温度值, 精确到小数点后一位
139技术分享图片void display(unsigned int v) 
140技术分享图片{
141技术分享图片    unsigned char count;
142技术分享图片    unsigned char datas[] = {0, 0, 0};
143技术分享图片    datas[0] = v / 100;
144技术分享图片    datas[1] = v % 100 / 10;
145技术分享图片    datas[2] = v % 10;
146技术分享图片    for(count = 0; count < 3; count++)
147技术分享图片    {
148技术分享图片        //片选
149技术分享图片        wela = 0; 
150技术分享图片        P0 = ((0xfe << count) | (0xfe >> (8 - count))); //选择第(count + 1) 个数码管
151技术分享图片        wela = 1; //打开锁存, 给它一个下降沿量
152技术分享图片        wela = 0;
153技术分享图片        //段选
154技术分享图片        dula = 0;
155技术分享图片        if(count != 1)
156技术分享图片        {
157技术分享图片            P0 = table[datas[count]];  //显示数字
158技术分享图片        }
159技术分享图片        else
160技术分享图片        {
161技术分享图片            P0 = tableWidthDot[datas[count]]; //显示带小数点数字
162技术分享图片        }
163技术分享图片        dula = 1;  //打开锁存, 给它一个下降沿量
164技术分享图片        dula = 0;
165技术分享图片        delay(5); //延时5ms, 即亮5ms
166技术分享图片
167技术分享图片        //清除段先, 让数码管灭, 去除对下一位的影响, 去掉高位对低位重影
168技术分享图片        //若想知道影响效果如何, 可自行去掉此段代码
169技术分享图片        //因为数码管是共阴极的, 所有灭的代码为: 00H
170技术分享图片        dula = 0;
171技术分享图片        P0 = 0x00;  //显示数字
172技术分享图片        dula = 1; //打开锁存, 给它一个下降沿量
173技术分享图片        dula = 0;
174技术分享图片    }
175技术分享图片}
176技术分享图片
177技术分享图片void main()
178技术分享图片{
179技术分享图片    unsigned char i;
180技术分享图片    unsigned int value;
181技术分享图片    while(1)
182技术分享图片    {
183技术分享图片        //启动温度转换
184技术分享图片        sendChangeCmd();
185技术分享图片        value = getTmpValue();
186技术分享图片        //显示3次
187技术分享图片        for(i = 0; i < 3; i++)
188技术分享图片        {
189技术分享图片            display(value);
190技术分享图片        }
191技术分享图片    }
192技术分享图片}

显示效果:
技术分享图片 


流程图:
技术分享图片

改进代码: 扩大测量范围, 使可测量范围为: -55度 ~ +125度, 严格按照上面的流程进行软件设计
3.15 1:34 修正display()函数中的下一位显示对上一位的影响
技术分享图片改进代码
  1技术分享图片#include <reg51.H>
  2技术分享图片#include<intrins.h>
  3技术分享图片#include <math.H>  //要用到取绝对值函数abs()
  4技术分享图片//通过DS18B20测试当前环境温度, 并通过数码管显示当前温度值, 目前显示范围: -55~ +125度
  5技术分享图片sbit wela = P2^7;  //数码管位选
  6技术分享图片sbit dula = P2^6;  //数码管段选
  7技术分享图片sbit ds = P2^2;
  8技术分享图片int tempValue;
  9技术分享图片
 10技术分享图片//0-F数码管的编码(共阴极)
 11技术分享图片unsigned char code table[]={0x3f,0x06,0x5b,0x4f,0x66,
 12技术分享图片    0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};
 13技术分享图片//0-9数码管的编码(共阴极), 带小数点
 14技术分享图片unsigned char code tableWidthDot[]={0xbf, 0x86, 0xdb, 0xcf, 0xe6, 0xed, 0xfd, 
 15技术分享图片    0x87, 0xff, 0xef};
 16技术分享图片
 17技术分享图片//延时函数, 对于11.0592MHz时钟, 例i=10,则大概延时10ms.
 18技术分享图片void delay(unsigned int i)
 19技术分享图片{
 20技术分享图片    unsigned int j;
 21技术分享图片    while(i--)
 22技术分享图片    {
 23技术分享图片        for(j = 0; j < 125; j++);
 24技术分享图片    }
 25技术分享图片}
 26技术分享图片
 27技术分享图片//初始化DS18B20
 28技术分享图片//让DS18B20一段相对长时间低电平, 然后一段相对非常短时间高电平, 即可启动
 29技术分享图片void dsInit()
 30技术分享图片{
 31技术分享图片    //对于11.0592MHz时钟, unsigned int型的i, 作一个i++操作的时间大于为8us
 32技术分享图片    unsigned int i;  
 33技术分享图片    ds = 0;
 34技术分享图片    i = 100;   //拉低约800us, 符合协议要求的480us以上
 35技术分享图片    while(i>0) i--;
 36技术分享图片    ds = 1;    //产生一个上升沿, 进入等待应答状态
 37技术分享图片    i = 4;
 38技术分享图片    while(i>0) i--;
 39技术分享图片}
 40技术分享图片
 41技术分享图片void dsWait()
 42技术分享图片{
 43技术分享图片     unsigned int i;
 44技术分享图片     while(ds);  
 45技术分享图片     while(~ds);  //检测到应答脉冲
 46技术分享图片     i = 4;
 47技术分享图片     while(i > 0) i--;
 48技术分享图片}
 49技术分享图片
 50技术分享图片//向DS18B20读取一位数据
 51技术分享图片//读一位, 让DS18B20一小周期低电平, 然后两小周期高电平, 
 52技术分享图片//之后DS18B20则会输出持续一段时间的一位数据
 53技术分享图片bit readBit()
 54技术分享图片{
 55技术分享图片    unsigned int i;
 56技术分享图片    bit b;
 57技术分享图片    ds = 0;
 58技术分享图片    i++;   //延时约8us, 符合协议要求至少保持1us
 59技术分享图片    ds = 1; 
 60技术分享图片    i++; i++;  //延时约16us, 符合协议要求的至少延时15us以上
 61技术分享图片    b = ds;
 62技术分享图片    i = 8; 
 63技术分享图片    while(i>0) i--;  //延时约64us, 符合读时隙不低于60us要求
 64技术分享图片    return b;
 65技术分享图片}
 66技术分享图片
 67技术分享图片//读取一字节数据, 通过调用readBit()来实现
 68技术分享图片unsigned char readByte()
 69技术分享图片{
 70技术分享图片    unsigned int i;
 71技术分享图片    unsigned char j, dat;
 72技术分享图片    dat = 0;
 73技术分享图片    for(i=0; i<8; i++)
 74技术分享图片    {
 75技术分享图片        j = readBit();
 76技术分享图片        //最先读出的是最低位数据
 77技术分享图片        dat = (j << 7) | (dat >> 1);
 78技术分享图片    }
 79技术分享图片    return dat;
 80技术分享图片}
 81技术分享图片
 82技术分享图片//向DS18B20写入一字节数据
 83技术分享图片void writeByte(unsigned char dat)
 84技术分享图片{
 85技术分享图片    unsigned int i;
 86技术分享图片    unsigned char j;
 87技术分享图片    bit b;
 88技术分享图片    for(j = 0; j < 8; j++)
 89技术分享图片    {
 90技术分享图片        b = dat & 0x01;
 91技术分享图片        dat >>= 1;
 92技术分享图片        //写"1", 将DQ拉低15us后, 在15us~60us内将DQ拉高, 即完成写1
 93技术分享图片        if(b)   
 94技术分享图片        {
 95技术分享图片            ds = 0;
 96技术分享图片            i++; i++;  //拉低约16us, 符号要求15~60us内
 97技术分享图片            ds = 1;    
 98技术分享图片            i = 8; while(i>0) i--;  //延时约64us, 符合写时隙不低于60us要求
 99技术分享图片        }
100技术分享图片        else  //写"0", 将DQ拉低60us~120us
101技术分享图片        {
102技术分享图片            ds = 0;
103技术分享图片            i = 8; while(i>0) i--;  //拉低约64us, 符号要求
104技术分享图片            ds = 1;
105技术分享图片            i++; i++;  //整个写0时隙过程已经超过60us, 这里就不用像写1那样, 再延时64us了
106技术分享图片        }
107技术分享图片    }
108技术分享图片}
109技术分享图片
110技术分享图片//向DS18B20发送温度转换命令
111技术分享图片void sendChangeCmd()
112技术分享图片{
113技术分享图片    dsInit();    //初始化DS18B20, 无论什么命令, 首先都要发起初始化
114技术分享图片    dsWait();   //等待DS18B20应答
115技术分享图片    delay(1);    //延时1ms, 因为DS18B20会拉低DQ 60~240us作为应答信号
116技术分享图片    writeByte(0xcc); //写入跳过序列号命令字 Skip Rom
117技术分享图片    writeByte(0x44); //写入温度转换命令字 Convert T
118技术分享图片}
119技术分享图片
120技术分享图片//向DS18B20发送读取数据命令
121技术分享图片void sendReadCmd()
122技术分享图片{
123技术分享图片    dsInit();
124技术分享图片    dsWait();
125技术分享图片    delay(1);
126技术分享图片    writeByte(0xcc); //写入跳过序列号命令字 Skip Rom
127技术分享图片    writeByte(0xbe); //写入读取数据令字 Read Scratchpad
128技术分享图片}
129技术分享图片
130技术分享图片//获取当前温度值
131技术分享图片int getTmpValue()
132技术分享图片{
133技术分享图片    unsigned int tmpvalue;
134技术分享图片    int value; //存放温度数值
135技术分享图片    float t;
136技术分享图片    unsigned char low, high;
137技术分享图片    sendReadCmd();
138技术分享图片    //连续读取两个字节数据
139技术分享图片    low = readByte(); 
140技术分享图片    high = readByte();
141技术分享图片    //将高低两个字节合成一个整形变量
142技术分享图片    //计算机中对于负数是利用补码来表示的
143技术分享图片    //若是负值, 读取出来的数值是用补码表示的, 可直接赋值给int型的value
144技术分享图片    tmpvalue = high;
145技术分享图片    tmpvalue <<= 8;
146技术分享图片    tmpvalue |= low;
147技术分享图片    value = tmpvalue;
148技术分享图片    
149技术分享图片    //使用DS18B20的默认分辨率12位, 精确度为0.0625度, 即读回数据的最低位代表0.0625度
150技术分享图片    t = value * 0.0625;
151技术分享图片    //将它放大100倍, 使显示时可显示小数点后两位, 并对小数点后第三进行4舍5入
152技术分享图片    //如t=11.0625, 进行计数后, 得到value = 1106, 即11.06 度
153技术分享图片    //如t=-11.0625, 进行计数后, 得到value = -1106, 即-11.06 度
154技术分享图片    value = t * 100 + (value > 0 ? 0.5 : -0.5); //大于0加0.5, 小于0减0.5
155技术分享图片    return value;
156技术分享图片}
157技术分享图片
158技术分享图片unsigned char const timeCount = 3; //动态扫描的时间间隔
159技术分享图片//显示当前温度值, 精确到小数点后一位
160技术分享图片//若先位选再段选, 由于IO口默认输出高电平, 所以当先位选会使数码管出现乱码
161技术分享图片void display(int v) 
162技术分享图片{
163技术分享图片    unsigned char count;
164技术分享图片    unsigned char datas[] = {0, 0, 0, 0, 0};
165技术分享图片    unsigned int tmp = abs(v);
166技术分享图片    datas[0] = tmp / 10000;
167技术分享图片    datas[1] = tmp % 10000 / 1000;
168技术分享图片    datas[2] = tmp % 1000 / 100;
169技术分享图片    datas[3] = tmp % 100 / 10;
170技术分享图片    datas[4] = tmp % 10;
171技术分享图片    if(v < 0)
172技术分享图片    {
173技术分享图片        //关位选, 去除对上一位的影响
174技术分享图片        P0 = 0xff; 
175技术分享图片        wela = 1; //打开锁存, 给它一个下降沿量
176技术分享图片        wela = 0;
177技术分享图片        //段选
178技术分享图片        P0 = 0x40; //显示"-"号
179技术分享图片        dula = 1;  //打开锁存, 给它一个下降沿量
180技术分享图片        dula = 0;
181技术分享图片
182技术分享图片        //位选
183技术分享图片        P0 = 0xfe; 
184技术分享图片        wela = 1; //打开锁存, 给它一个下降沿量
185技术分享图片        wela = 0;
186技术分享图片        delay(timeCount); 
187技术分享图片    }
188技术分享图片    for(count = 0; count != 5; count++)
189技术分享图片    {
190技术分享图片        //关位选, 去除对上一位的影响
191技术分享图片        P0 = 0xff; 
192技术分享图片        wela = 1; //打开锁存, 给它一个下降沿量
193技术分享图片        wela = 0;
194技术分享图片        //段选
195技术分享图片        if(count != 2)
196技术分享图片        {
197技术分享图片        /*    if((count == 0 && datas[count] == 0) 
198技术分享图片                || ((count == 1 && datas[count] == 0) && (count == 0 && datas[count - 1] == 0)))
199技术分享图片            {
200技术分享图片                P0 = 0x00; //当最高位为0时, 不作显示
201技术分享图片            }
202技术分享图片            else*/
203技术分享图片                P0 = table[datas[count]];  //显示数字
204技术分享图片        }
205技术分享图片        else
206技术分享图片        {
207技术分享图片            P0 = tableWidthDot[datas[count]]; //显示带小数点数字
208技术分享图片        }
209技术分享图片        dula = 1;  //打开锁存, 给它一个下降沿量
210技术分享图片        dula = 0;
211技术分享图片
212技术分享图片        //位选 
213技术分享图片        P0 = _crol_(0xfd, count); //选择第(count + 1) 个数码管
214技术分享图片        wela = 1; //打开锁存, 给它一个下降沿量
215技术分享图片        wela = 0;
216技术分享图片        delay(timeCount); 
217技术分享图片    }
218技术分享图片}
219技术分享图片
220技术分享图片void main()
221技术分享图片{
222技术分享图片    unsigned char i;
223技术分享图片    
224技术分享图片    while(1)
225技术分享图片    {
226技术分享图片        //启动温度转换
227技术分享图片        sendChangeCmd();
228技术分享图片        //显示5次
229技术分享图片        for(i = 0; i < 40; i++)
230技术分享图片        {
231技术分享图片            display(tempValue);
232技术分享图片        }
233技术分享图片        tempValue = getTmpValue();
234技术分享图片    }
235技术分享图片}

改进后的效果图:
只有一位小数
技术分享图片 
两位小数,  并消除下一位对上一位的影响
技术分享图片








































































































































































































































































































































































































































































































































以上是关于单片机练习 - DS18B20温度转换与显示的主要内容,如果未能解决你的问题,请参考以下文章

Proteus51单片机+双DS18B20浮点数温度显示

单片机上的ds18b20温度传感器是啥工作原理

STC单片机LCD1602四线驱动模式显示DS18B20温度

STC单片机硬件I2C驱动ssd1306 OLED显示DS18B20温度

16.STC15W408AS单片机获取DS18B20温度

16.STC15W408AS单片机获取DS18B20温度