当前位置:首页 > 单片机 > 单片机
[导读] DS18B20是最常用来学习某一个新的开发工具的,程序都是大同小异,主要是要注意时序中的延时要准确,指令要正确,这里记录一下!------------------第一部分是--------ds18b20.h----------------------#

DS18B20是最常用来学习某一个新的开发工具的,程序都是大同小异,主要是要注意时序中的延时要准确,指令要正确,这里记录一下!


------------------第一部分是--------ds18b20.h----------------------

#ifndef __DS18B20_H
#define __DS18B20_H

#include "stm32f10x.h"
#include "bsp_SysTick.h" //精确延时函数头文件----参考http://blog.csdn.net/xuxuechen/article/details/40783209这个看一下

#define HIGH 1
#define LOW 0

#define DS18B20_CLK RCC_APB2Periph_GPIOB
#define DS18B20_PIN GPIO_Pin_10
#define DS18B20_PORT GPIOB //总体代表DS18B20的GPIO口为PB10


//带参宏,可以像内联函数一样使用,输出高电平或低电平
#define DS18B20_DATA_OUT(a)if (a)
GPIO_SetBits(GPIOB,GPIO_Pin_10);
else
GPIO_ResetBits(GPIOB,GPIO_Pin_10)
//读取引脚的电平
#define DS18B20_DATA_IN() GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_10)

uint8_t DS18B20_Init(void);
float DS18B20_Get_Temp(uint8_t *a,uint8_t b);
void read_serial(uint8_t *serial);
#endif /* __DS18B20_H */

----------------第二部分是----------DS18B20.C源文件---------------------


#include "bsp_ds18b20.h"

uint8_t serial_1[8]={0x28,0x2d,0x9a,0xdd,0x02,0x00,0x00,0x3b};
uint8_t serial_2[8]={0x28,0x3b,0x2b,0xbc,0x02,0x00,0x00,0x4f};
uint8_t serial_3[8]={0x28,0x00,0x49,0x1b,0x03,0x00,0x00,0x4c};//序列号,需要根据自己的DS18B20修改,具体读取方法,会有另一篇介绍。

static void DS18B20_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(DS18B20_CLK, ENABLE);
GPIO_InitStructure.GPIO_Pin = DS18B20_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(DS18B20_PORT, &GPIO_InitStructure);

GPIO_SetBits(DS18B20_PORT, DS18B20_PIN);
}

static void DS18B20_Mode_IPU(void) //使DS18B20-DATA引脚变为输入模式
{
GPIO_InitTypeDef GPIO_InitStructure;

GPIO_InitStructure.GPIO_Pin = DS18B20_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
GPIO_Init(DS18B20_PORT, &GPIO_InitStructure);
}

static void DS18B20_Mode_Out_PP(void) //使DS18B20-DATA引脚变为输出模式
{
GPIO_InitTypeDef GPIO_InitStructure;

GPIO_InitStructure.GPIO_Pin = DS18B20_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(DS18B20_PORT, &GPIO_InitStructure);
}

static void DS18B20_Rst(void) //主机给从机发送复位脉冲
{
DS18B20_Mode_Out_PP();
DS18B20_DATA_OUT(LOW);
Delay_Us(750); //主机至少产生480us的低电平复位信号
DS18B20_DATA_OUT(HIGH); //主机在产生复位信号后,需将总线拉高
Delay_Us(15); //从机接收到主机的复位信号后,会在15~60us后给主机发一个存在脉冲
}

static uint8_t DS18B20_Presence(void) //检测从机给主机返回的存在脉冲
{
uint8_t pulse_time = 0;

DS18B20_Mode_IPU(); //主机设置为上拉输入

//等待存在脉冲的到来,存在脉冲为一个60~240us的低电平信号
//如果存在脉冲没有来则做超时处理,从机接收到主机的复位信号后,会在15~60us后给主机发一个存在脉冲

while( DS18B20_DATA_IN() && pulse_time<100 )
{
pulse_time++;
Delay_Us(1);
}
if( pulse_time >=100 ) //经过100us后,存在脉冲都还没有到来
return 1;
else
pulse_time = 0;
while( !DS18B20_DATA_IN() && pulse_time<240 ) //存在脉冲到来,且存在的时间不能超过240us
{
pulse_time++;
Delay_Us(1);
}
if( pulse_time >=240 )
return 1;
else
return 0;
}

static uint8_t DS18B20_Read_Bit(void) //从DS18B20读取一个bit
{
uint8_t dat;

DS18B20_Mode_Out_PP(); //读0和读1的时间至少要大于60us
DS18B20_DATA_OUT(LOW); //读时间的起始:必须由主机产生 >1us <15us 的低电平信号
Delay_Us(10);
DS18B20_Mode_IPU(); //设置成输入,释放总线,由外部上拉电阻将总线拉高
if( DS18B20_DATA_IN() == SET )
dat = 1;
else
dat = 0;
Delay_Us(45); //这个延时参数请参考时序图

return dat;
}

uint8_t DS18B20_Read_Byte(void) //从DS18B20读一个字节,低位先行
{
uint8_t i, j, dat = 0;

for(i=0; i<8; i++)
{
j = DS18B20_Read_Bit();
dat = (dat) | (j< }

return dat;
}

void DS18B20_Write_Byte(uint8_t dat) //写一个字节到DS18B20,低位先行
{
uint8_t i, testb;

DS18B20_Mode_Out_PP();
for( i=0; i<8; i++ )
{
testb = dat&0x01;
dat = dat>>1;
if (testb)//写0和写1的时间至少要大于60us
{
DS18B20_DATA_OUT(LOW);
Delay_Us(8);
DS18B20_DATA_OUT(HIGH);
Delay_Us(58);
}
else
{
DS18B20_DATA_OUT(LOW);
Delay_Us(70);
DS18B20_DATA_OUT(HIGH);
Delay_Us(2);
}
}
}

void DS18B20_Start(void)
{
DS18B20_Rst();
DS18B20_Presence();
DS18B20_Write_Byte(0XCC); //跳过 ROM
DS18B20_Write_Byte(0X44); //开始转换
}

uint8_t DS18B20_Init(void)
{
DS18B20_GPIO_Config();
DS18B20_Rst();

return DS18B20_Presence();
}

void DS18B20_Match_Serial(uint8_t a) //匹配序列号
{
uint8_t i;
DS18B20_Rst();
DS18B20_Presence();
DS18B20_Write_Byte(0X55); //匹配序列号指令
if(a==1)
{
for(i=0;i<8;i++)
DS18B20_Write_Byte(serial_1[i]);
}
else if(a==2)
{
for(i=0;i<8;i++)
DS18B20_Write_Byte(serial_2[i]);
}
else if(a==3)
{
for(i=0;i<8;i++)
DS18B20_Write_Byte(serial_3[i]);
}
}


//存储的温度是16 位的带符号扩展的二进制补码形式
//当工作在12位分辨率时,其中5个符号位,7个整数位,4个小数位

// |---------整数----------|-----小数 分辨率 1/(2^4)=0.0625----|
//低字节 | 2^3 | 2^2 | 2^1 | 2^0 | 2^(-1) | 2^(-2) | 2^(-3) | 2^(-4) |

// |-----符号位:0->正 1->负-------|-----------整数-----------|
//高字节 | s | s | s | s | s | 2^6 | 2^5 | 2^4 |
//温度 = 符号位 + 整数 + 小数*0.0625


float DS18B20_Get_Temp(uint8_t*a,uint16_tgo_temp,uint8_tb)
{
uint8_t tpmsb, tplsb;
short s_tem;
float f_tem;
int temp_num;

DS18B20_Rst();
DS18B20_Presence();
DS18B20_Write_Byte(0XCC); // 跳过 ROM
DS18B20_Match_Serial(b); //匹配序列号
DS18B20_Write_Byte(0X44); // 开始转换

DS18B20_Rst();
DS18B20_Presence();
DS18B20_Write_Byte(0XCC); //跳过 ROM
DS18B20_Match_Serial(b); //匹配序列号
DS18B20_Write_Byte(0XBE); //读温度值

tplsb = DS18B20_Read_Byte();
tpmsb = DS18B20_Read_Byte();

s_tem = tpmsb<<8;
s_tem = s_tem | tplsb;

if( s_tem < 0 ) //负温度
{
f_tem = (~s_tem+1) * 0.0625;
temp_num = (~s_tem+1) * 0.0625*10;
go_temp= temp_num;
if(temp_num>=1000)
{
a[0]='-';
a[1]= temp_num/1000+'0';
a[2]= temp_num%1000/100+'0';
a[3]= temp_num%100/10+'0';
a[4]='.';
a[5]= temp_num%10+'0';
a[6]= '';
}
else
{
a[0]='-';
a[1]= temp_num/100+'0';
a[2]= temp_num%100/10+'0';
a[3]='.';
a[4]=temp_num%10+'0';
a[5]= '';
}
}
else
{
f_tem = s_tem * 0.0625;
temp_num = s_tem * 0.0625*10;
go_temp= temp_num;
if(temp_num>=1000)
{
a[0]='+';
a[1]= temp_num/1000+'0';
a[2]= temp_num%1000/100+'0';
a[3]= temp_num%100/10+'0';
a[4]='.';
a[5]= temp_num%10+'0';
a[6]= '';
}
else
{
a[0]='+';
a[1]= temp_num/100+'0';
a[2]= temp_num%100/10+'0';
a[3]='.';
a[4]=temp_num%10+'0';
a[5]= '';
}
}
return f_tem;
}

void read_serial(uint8_t *serial) //读取序列号
{
uint8_t i;
DS18B20_Rst();
DS18B20_Presence();
DS18B20_Write_Byte(0X33); //读取序列号指令
for(i=0;i<8;i++)
serial[i] = DS18B20_Read_Byte();
}

----------------第三部分是-------------main.c--------------

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

阿联酋迪拜2025年8月26日 /美通社/ -- 纳斯达克上市公司Robo.ai Inc.今日正式宣布完成品牌焕新升级,并于8月26日正式启用全新纳斯达克股票代码"...

关键字: AI 人工智能 代码 智能科技

北京2025年8月13日 /美通社/ -- 近日,北京积算科技有限公司(以下简称"积算科技")推出一站式AlphaFold3在线算力服务,现已开放免费使用。其内置优化后的AlphaFold3模型,支持...

关键字: ALPHA 代码 图形化 蛋白质

北京2025年7月21日 /美通社/ -- 浪潮信息宣布元脑企智一体机已率先完成对Kimi K2 万亿参数大模型的适配支持,并实现单用户70 tokens/s的流畅输出速度,为企业客户高效部署应用大模型提供高处...

关键字: 模型 AGENT TOKEN 代码

7月18日,一则“微信安卓安装包出现5处fxxk”的话题,迅速登上微博热搜,吸引了众多网友的热议和关注。

关键字: 代码 程序员

共鉴AI未来,缅怀先辈贡献 深圳 2025年5月21日 /美通社/ -- 5月16日下午,深圳市金澄智创AI+传承迎来了乔迁之喜,一场意义非凡的活动在新址盛大举行。活动现场星光熠熠,庄世平前辈之子庄荣新先生、南方财经...

关键字: AI AI技术 BSP 代码

C语言代码优化与性能提升是软件开发中至关重要的一环。优化C语言代码不仅可以提高程序的执行效率,还可以减少资源消耗,提升用户体验。

关键字: C语言 代码

Qt 路线图致力于实现与各行业技术栈的无缝集成,助力企业与开发者使用自选工具,更高效地构建、扩展和维护软件解决方案。 芬兰埃斯波 2025年5月7日 /美通社/ -- 在今日举行的2025年Qt全球峰会上,Qt...

关键字: GROUP 生态系统 开发者 代码

——首批"专家级数字员工"部署上岗,构建组织传承与人才战略的AI导师矩阵 北京2025年4月25日 /美通社/ -- 在生成式AI技术加速重构组织能力与人才机制的当下,如何留住专家经验、培养管理人才...

关键字: AI 智能体 矩阵 代码

深圳2025年4月21日 /美通社/ -- 近日,国家发展改革委公布第 31 批国家企业技术中心认定名单,华测检测认证集团股份有限公司(股票代码:300012.SZ,简称 "CTI 华测检测")凭借在检验检测领域的全链条...

关键字: TI IC 代码 半导体

珠海2025年3月13日 /美通社/ -- 近期,AI工具DeepSeek引发广泛关注,大量非专业投资者试图借助其生成股票策略,却面临现实困境:平台输出的Python代码策略对零编程基础用户存在极高使用门槛。即...

关键字: EPS 代码 PSE AI
关闭