1.前言
我家在东北农村,冬天很冷,买了一个锅炉,需要循环泵的。简单来说就是锅炉水热了之后循环泵自动开启,然后将热水输送走,送到暖气,热水抽走,凉水进入锅炉,温度降低,循环泵关闭,等待下一次水烧热。因为需要取暖的房子距离烧锅炉的地方比较远,所以需要循环泵,如果距离近的话水烧热后利用热水上流冷水回流的原理会自动完成循环。当然目前市场上有这种利用温度自动控制循环泵开启关闭的设备:
原理就是有一个热敏电阻探头(带有磁铁吸附,可以吸附到锅炉壁上),然后一个继电器控制的。当温度达到设定的值后,继电器开启,循环泵启动,循环过后锅炉壁温度下降,继电器关闭,循环泵关闭。
循环泵:
由于我家的循环泵功率较大,小继电器启动几次就烧坏了,所以中间又接入了一层交流接触器(我的老父亲加的)。
这种市面上的设备可以大致解决水循环的问题,但是也有一些细节问题解决不了,例如:当回流管水温度也达到设定的温度值后,循环泵就会一直开启状态,这时候就需要手动去调节旋钮温度(调高),让循环泵停下来(大功率循环泵很耗电)。又如 当炉子内部煤渐渐烧结束后(东北话叫涝了),锅炉温度达不到设定值,这个时候就需要降低旋钮,让循环泵启动,让这些煤产生的余温不浪费掉。
由于以上的不足,就需要有人隔几个小时去调节旋钮。这部分工作都是我老父亲在做,所以我想做一个自动的东西降低一下老父亲的工作量。
2.项目结构
我最初的构思是通过手机可以远程调节温度,这样至少不用手动去调节旋钮了,我半夜起来在被窝里用手机调节一下就可以了。基于最初的构思设计的结构:
3.硬件搭建
esp01模块+继电器模块,220v转5v模块 + 插排 = 联网插座
esp8226 + 温度传感器 + 数码管 = 实时温度检测显示联网模块
我的控制系统都写在 esp8226中。
温度传感器探头制作:要驱动ds18b20传感器,需要在数据总线(DQ)与 VDD 引脚之间加入一个4.7k欧姆的电阻(上拉电阻),这个是必须有的,用一个铁纽扣包裹起来,里面放了两个小块的钕磁铁,最后用哥俩好胶水灌满,这样就形成了一个温度传感器探头,可以吸附在锅炉壁上:
这一步我犯了一个错误,将ds18b20传感器和上拉电阻都放入到探头里面,测温的时候发现温度升高以后温度传感器读取温度失败,是因为温度升高导致电阻阻值增大。后来我将其电阻放到尾部就解决了这个问题。
封装后的探头:
esp8226 + 数码管模块:
壳子是我用3D打印机打印的,里面放着esp8226模块。
4.软件编写
固件
esp8226 和 esp01 我都是烧录的micropython固件。
通信
通信开始我想的是用socket实现,简单弄个HTTP协议进行通信,但是手机上这样控制是比较麻烦的,没有好用的软件,我还要做个web界面,另外socket会阻塞线程。
后来我发现了MQTT这种协议,这是一种针对物联网的硬件网络通信协议,可以应对高延迟的网络环境。简单介绍一下MQTT协议,首先需要一个服务端(也有叫代理服务器的,运行在电脑上,这里我运行在树莓派上),然后你的所有物联网硬件设备(如esp8226)都是客户端,物联网硬件设备(客户端)之间不会直接进行通信,它们都是与服务端直接进行通信,硬件设备之间如果需要进行通信是需要发布或者订阅主题的。例如:如果硬件A 和硬件B 之间需要进行通信,那么首先它们都需要连接到 服务端,然后 A 发布一个名为topic的主题,如果硬件B要接受A的信息就需要订阅 topic这个主题,这样就实现了 A ---> B 的通信,那么反向通信也是一个原理,B 发布主题,A来订阅。这个协议还有一个优势,可以一个主题多个客户端去订阅,这样就能实现多端通信。mqtt协议通信中每个客户端不知道其它客户端的存在,他们都是与服务端进行直接通信。
mqtt协议在micropython固件(我不知道安信可默认固件是不是也有)通信有一个很大的问题:报错。例如,如果你在开始的时候订阅了一个主题,然后你使用非阻塞的方法通过回调函数处理 订阅的主题消息,这个时候如果你订阅的主题发布的次数超过你 的次数,那么就会在固件底层报一个错误 ,然后通信中断,简单来说就是你订阅一个主题后进入事件循环,来不及处理订阅主题的消息,就会导致这个错误。这个错误官方解释是tcp buffer资源没有释放导致的,因为esp8226可用资源非常有限,但是在mqtt通信中我不可能每次循环后都去释放连接,然后每次都重新建立连接,并且这个报错恶心的地方在于, 是捕获不到的,那么对于处理这个错误只能通过设计上来解决了,也就是一开始就保证其订阅的主题发布的间隔远小于其循环的间隔,保证每次都能及时的处理订阅的消息。这个问题我解决了很久,因为 捕获不到,并且不常出现,无法定位,完全不知道为什么通信会中断。
代码核心逻辑编写
最核心的控制逻辑我都写在模块中,循环采集温度,然后将温度作为一个主题发布出来,手机可以订阅主题就可以实时查看锅炉温度了,在循环中有一个的变量 ,如果采集的温度高于 那么就发布一个 开启的主题,如果温度低于 ,那么就发布一个 关闭,并且每次订阅 设置主题,用于修改变量。
esp01 控制继电器,每次订阅 就可以了,然后每次再把 作为一个主题发布出来。
这个代码逻辑最开始我是这样写的,手机上只需要发布设置温度主题就可以了,但是也是需要人一段时间用手机去调节一次,还是不能实现自动化,后来我又修改的 esp8226 中的代码逻辑,复杂了一些。
esp8226 核心逻辑还是和上述一样,每次检测温度高于设定温度后还会开启开关,但是启动后检测到温度低于设定温度不会立刻关闭开关,而是等待检测温度低于设定温度减去一个变量再发布一个开关关闭的主题。这是为了解决 检测温度在设定温度临界反复跳变,从而导致开关在短时间内反复的开启关闭,这里引入了一个减去变量,我将其叫做 。这里就有一个问题,如果锅炉一直在升温,那么就算循环泵一直开启 ,锅炉温度也不会低于设定温度(回流管温度已经高于设定温度了),那么循环泵就会处于一直开启状态,所以这里我引入了第二个变量:,每次启动后我开启一个计时器,如果计时器时间超过 ,那么无论此时的温度是否低于,都会关闭循环泵,并且将提高,提高温度为加当前温度,这里就实现了 自动调节 的(升高方向)。还有一个问题:当炉子煤燃烧殆尽的时候,温度逐渐降低,检测到的温度肯定会远低于设定温度,循环泵就永远不会开启了。所以这里我引入了第三个变量(),当esp8226 上电启动时,启动一个计时器,如果计时时间 等于,那么将 调节到当前温度,用于 自动回调,假设 为 30分钟,那么即使 高于 检测温度,也会实现30分钟一启动循环泵,和剩余逻辑实现一个闭环,这样就能实现 ,上下自动调节。 这里我还加入了一个变量,如果当前温度低于,那么就算 当前温度高于 也不会启动循环泵,用于没烧煤炭的时候,这样就防止了无论什么时候都30分钟启动一次循环泵, 设置成高于环境温度一些就可以了。
这里 ,,,,,都可以通过手机端进行设置,我让esp8226订阅这些主题用于设置这些变量。
esp8226代码:
```python from machine import Pin,reset import onewire from ds18x20 import DS18X20 import time import tm1637 from umqtt.simple import MQTTClient
def main(): client_id = "esp_temperature" mserver = '192.168.0.99' #mserver = '192.168.3.200' #mserver = 'mq.tongxinmao.com'
```
esp01中代码:
```python from machine import Pin from umqtt.simple import MQTTClient import time
def main(): def sub_callback(topic, msg): nonlocal client nonlocal pin """ 收到订阅消息回调 """ if msg == b'0': pin.off() else: pin.on() client.publish("switchWell",str(pin.value()),retain=True)
```
5.客户端监控调节软件
1.手机端:
MQTT Dash:
IoTMQTTPanel:
手机上我还是推荐,因为将一套面板的配置发布到另一台手机很方便,只需要发布一个主题,然后接收端订阅一个同名主题就可以把整个面板发布过去。 也具有这个功能,但是发布过去后会卡死,不知道为什么。
2.电脑端:
电脑端我还不知道有什么好用的软件,所以我用 PyQt5 简单写了一个监控的软件,没有调节功能,因为当时设备还不是很稳定,所以我一直在监控运行状态。