手把手教你用DS18B20玩转1-Wire单总线协议(附实测代码)
从零构建1-Wire通信系统基于DS18B20的实战指南1. 初识1-Wire协议与DS18B20第一次接触1-Wire协议时我被它的简洁性震撼到了——仅用一根数据线就能完成双向通信这听起来像是某种电子魔法。但当我真正把DS18B20温度传感器接入树莓派看到温度数据稳定输出时才理解这种单总线设计的精妙之处。1-Wire协议由Dallas Semiconductor现为Maxim Integrated子公司开发特别适合传感器网络、身份识别等低带宽应用场景。与I2C、SPI等多线协议相比它最大的优势就是布线简单——只需要一根数据线加上地线就能组建通信网络。这根线既传输时钟又传输数据还能通过寄生供电方式为从设备提供电能。DS18B20是1-Wire家族中最经典的数字温度传感器具有以下特性测量范围-55°C至125°C±0.5°C精度分辨率可调9至12位对应0.5°C至0.0625°C唯一64位地址每个传感器都有全球唯一标识多路复用能力单总线上可挂载多个设备# 示例读取DS18B20温度值的简化流程 1. 主机发送复位脉冲 2. DS18B20回应存在脉冲 3. 主机发送跳过ROM命令(0xCC) 4. 主机发送启动温度转换命令(0x44) 5. 等待转换完成(最多750ms) 6. 再次初始化总线 7. 发送读取暂存器命令(0xBE) 8. 读取9字节数据(包含温度值和CRC校验)2. 硬件连接与电路设计2.1 基础连接方案DS18B20支持两种供电模式外部供电和寄生供电。对于新手我强烈建议先从外部供电开始等熟悉协议后再尝试寄生供电模式。标准三线连接法外部供电DS18B20引脚说明 1. GND → 接电源地 2. DQ → 接MCU GPIO需4.7K上拉电阻 3. VDD → 接3.3V/5V电源注意即使使用外部供电数据线(DQ)也必须连接4.7KΩ上拉电阻这是保证信号完整性的关键。2.2 寄生供电的特殊考量当采用寄生供电时VDD引脚需要接地此时器件通过DQ线偷取电能。这种模式虽然节省了一根电源线但需要注意温度转换期间DQ线必须保持高电平以提供足够电能长距离传输时信号质量可能下降同时操作多个设备时可能出现供电不足// 寄生供电模式下强上拉的实现示例(STM32 HAL) void OneWire_StrongPullup(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) { HAL_GPIO_WritePin(GPIOx, GPIO_Pin, GPIO_PIN_SET); // 置高 GPIO_InitTypeDef GPIO_InitStruct {0}; GPIO_InitStruct.Pin GPIO_Pin; GPIO_InitStruct.Mode GPIO_MODE_OUTPUT_PP; // 推挽输出 GPIO_InitStruct.Speed GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOx, GPIO_InitStruct); }2.3 多设备组网方案1-Wire协议真正的威力在于单总线可以挂载多个设备。每个DS18B20都有唯一的64位ROM码格式如下字节位置内容说明0家族码(0x28)DS18B20的固定标识1-648位序列号厂商分配的全球唯一标识7CRC校验前56位的循环冗余校验搜索多个设备的算法采用二叉树遍历原理这里给出简化步骤发送复位脉冲并检测存在脉冲发送搜索ROM命令(0xF0)按位读取所有设备的响应会产生位冲突通过特定算法解析冲突逐步确定每个设备的完整ROM码3. 协议时序与底层驱动实现3.1 精确控制时序1-Wire协议对时序要求极为严格不同操作的时间窗口可能只有几微秒。以下是关键时序参数操作类型时间参数标准模式高速模式复位脉冲主机拉低时间480μs48μs从机响应时间15-60μs2-6μs写时隙写1拉低时间1-15μs1-2μs写0拉低时间60-120μs6-12μs读时隙主机拉低时间1-15μs1-2μs采样窗口时间15μs内2μs内; 示例AVR汇编实现的精确延时循环 Delay15us: ; 15微秒延时 16MHz ldi r24, 48 ; 1 cycle Loop: dec r24 ; 1 cycle brne Loop ; 2 cycles (1 when exits) ret ; 4 cycles3.2 完整驱动代码解析下面是一个基于STM32的1-Wire驱动实现包含核心操作函数// 复位脉冲与存在检测 uint8_t OneWire_Reset(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) { GPIO_InitTypeDef GPIO_InitStruct {0}; // 配置为开漏输出 GPIO_InitStruct.Pin GPIO_Pin; GPIO_InitStruct.Mode GPIO_MODE_OUTPUT_OD; GPIO_InitStruct.Pull GPIO_NOPULL; GPIO_InitStruct.Speed GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOx, GPIO_InitStruct); // 拉低480us HAL_GPIO_WritePin(GPIOx, GPIO_Pin, GPIO_PIN_RESET); Delay_us(480); // 释放总线切换为输入 GPIO_InitStruct.Mode GPIO_MODE_INPUT; HAL_GPIO_Init(GPIOx, GPIO_InitStruct); // 等待15-60us内的存在脉冲 Delay_us(60); uint8_t presence !HAL_GPIO_ReadPin(GPIOx, GPIO_Pin); // 等待剩余时间完成复位周期 Delay_us(420); return presence; } // 写入一个位 void OneWire_WriteBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin, uint8_t bit) { GPIO_InitTypeDef GPIO_InitStruct {0}; GPIO_InitStruct.Pin GPIO_Pin; GPIO_InitStruct.Mode GPIO_MODE_OUTPUT_OD; HAL_GPIO_Init(GPIOx, GPIO_InitStruct); HAL_GPIO_WritePin(GPIOx, GPIO_Pin, GPIO_PIN_RESET); if(bit) { Delay_us(5); // 写1保持5us低电平 HAL_GPIO_WritePin(GPIOx, GPIO_Pin, GPIO_PIN_SET); Delay_us(55); // 剩余时隙时间 } else { Delay_us(60); // 写0保持60us低电平 HAL_GPIO_WritePin(GPIOx, GPIO_Pin, GPIO_PIN_SET); Delay_us(5); // 恢复时间 } } // 读取一个位 uint8_t OneWire_ReadBit(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin) { uint8_t bit 0; GPIO_InitTypeDef GPIO_InitStruct {0}; GPIO_InitStruct.Pin GPIO_Pin; GPIO_InitStruct.Mode GPIO_MODE_OUTPUT_OD; HAL_GPIO_Init(GPIOx, GPIO_InitStruct); // 拉低2us启动读时隙 HAL_GPIO_WritePin(GPIOx, GPIO_Pin, GPIO_PIN_RESET); Delay_us(2); // 释放总线切换为输入 GPIO_InitStruct.Mode GPIO_MODE_INPUT; HAL_GPIO_Init(GPIOx, GPIO_InitStruct); // 在15us内采样 Delay_us(8); bit HAL_GPIO_ReadPin(GPIOx, GPIO_Pin); // 等待时隙结束 Delay_us(50); return bit; }4. 高级应用与性能优化4.1 温度采集策略优化当系统中有多个DS18B20时合理的采集策略可以大幅提高效率并行转换技术发送跳过ROM命令(0xCC)发送启动转换命令(0x44)所有传感器会同时开始温度转换转换期间可以处理其他任务分辨率选择权衡分辨率转换时间温度步进适用场景9位93.75ms0.5°C快速响应的控制系统10位187.5ms0.25°C一般环境监测11位375ms0.125°C实验室级测量12位750ms0.0625°C高精度应用# Python示例设置分辨率(树莓派DS18B20) def set_resolution(sensor_id, bits): with open(f/sys/bus/w1/devices/{sensor_id}/resolution, w) as f: f.write(str(bits)) # 设置为11位分辨率(0.125°C精度) set_resolution(28-011931b2d0ee, 11)4.2 抗干扰与长距离传输当1-Wire总线长度超过10米时需要考虑信号完整性问题。以下是几个实用技巧使用屏蔽双绞线将DQ线与GND双绞减少电磁干扰降低上拉电阻值长距离时可尝试3.3KΩ甚至2.2KΩ增加总线驱动如DS2480B等专用线路驱动器分段供电每5-10米设置一个电源注入点提示判断信号质量的最简单方法是观察复位脉冲后的存在检测响应。如果时有时无通常表明线路阻抗不匹配或干扰过大。4.3 电源管理与低功耗设计对于电池供电的物联网设备电源优化至关重要间歇工作模式每小时唤醒一次采集温度采集后立即进入深度睡眠利用DS18B20的报警搜索功能(0xEC)实现事件触发供电方案对比方案优点缺点外部供电稳定可靠需要额外电源线寄生供电两线制简化布线转换期间需保持强上拉超级电容缓冲平衡功耗与布线增加BOM成本和体积// 低功耗示例使用STM32的STOP模式与RTC唤醒 void enter_low_power_mode(uint32_t sleep_seconds) { // 配置RTC唤醒中断 HAL_RTCEx_SetWakeUpTimer_IT(hrtc, sleep_seconds * 8, RTC_WAKEUPCLOCK_RTCCLK_DIV16); // 关闭外设时钟 __HAL_RCC_GPIOA_CLK_DISABLE(); __HAL_RCC_GPIOB_CLK_DISABLE(); // 进入STOP模式 HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI); // 唤醒后重新初始化系统时钟 SystemClock_Config(); }5. 故障排查与实战经验5.1 常见问题诊断表现象可能原因解决方案无法检测到设备接线错误/接触不良检查GND连接重焊连接点上拉电阻缺失或阻值不对添加/更换4.7KΩ上拉电阻温度读数固定为85°C电源不稳导致转换失败加强电源滤波检查VDD电压时序不符合规范用逻辑分析仪验证时序多设备时数据混乱ROM码冲突或搜索算法错误重新实现ROM搜索算法CRC校验频繁失败线路干扰或信号反射缩短总线长度加屏蔽5.2 逻辑分析仪调试技巧使用Saleae Logic或DSView等工具分析1-Wire信号时注意以下要点设置合适的采样率至少4MHz最好10MHz以上添加协议解码器大多数分析软件都有内置1-Wire解码关键检查点复位脉冲宽度(480μs±10%)存在脉冲的响应时间(15-60μs)位时隙的严格时序(写0/1的不同低电平时间)# 使用Linux内核的1-Wire子系统调试 # 查看已连接的1-Wire设备 ls /sys/bus/w1/devices/ # 监控原始数据流(需要root权限) cat /sys/bus/w1/devices/28-*/w1_slave5.3 温度校准与补偿虽然DS18B20出厂时已经校准但在极端环境或高精度需求下可以考虑两点校准法在冰水混合物(0°C)中记录读数在沸水(100°C考虑海拔修正)中记录读数计算偏移量和增益系数非线性补偿# 示例二阶温度补偿算法 def compensate_temp(raw_temp, cal_params): a, b, c cal_params # 通过校准获得的系数 return a * raw_temp**2 b * raw_temp c热惯性处理连续采样5次去掉最高最低值后取平均对快速变化的环境采用滑动窗口滤波在最近的一个温室监控项目中我们发现DS18B20在阳光直射下读数偏高2-3°C。通过给传感器加装防辐射罩并在软件中增加阳光照射时段的补偿系数最终将误差控制在±0.5°C以内。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2429070.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!