使用MQTT Python发布和订阅两种方式

时间:2018-06-15 17:03:45

标签: python mqtt

我目前在Raspberry Pi 3上编写了一个Python程序来读取湿度和温度传感器数据,并将这些数据发布到一个主题。然后我可以使用笔记本电脑接收这些数据。这是我的代码读取传感器数据并将其发布到我的Raspberry Pi中的主题

import RPi.GPIO as GPIO
import time
import json
import Adafruit_DHT as dht
import math
import paho.mqtt.publish as publish
import paho.mqtt.client as mqtt
# Creating the JSON Objects

dht22 = {}
arduino = {}
dht22Temp = []
dht22Hum = []
arduinoLED = []


dht22['temperature'] = dht22Temp
dht22['humidity'] = dht22Hum
dht22['sensor'] = 'DHT22'

arduino['blink'] = arduinoLED
arduino['actuator'] = 'arduinoLED'  

# Timing constants
E_PULSE = 0.0005
E_DELAY = 0.0005

def main():
  # Main program block
  while True:

    h, t = dht.read_retry(dht.DHT22, 17) //Reading humidity and temp data from GPIO17
    t = round(t,2)
    h = round(h,2)

    if t > 25: 
      if len(arduinoLED) == 3:
        arduinoLED.pop(0)
        arduinoLED.append("true")
      else: 
        arduinoLED.append("true")
    else:
      if len(arduinoLED) == 3:
        arduinoLED.pop(0)
        arduinoLED.append("false")
      else: 
        arduinoLED.append("false")
    if len(dht22Temp) == 3:
      dht22Temp.pop(0)
      dht22Temp.append(t)
    else: 
      dht22Temp.append(t)
    if len(dht22Hum) == 3:
      dht22Hum.pop(0)
      dht22Hum.append(h)
    else: 
      dht22Hum.append(h)
    # lm35dzTemp.append(tempc) 


    # Publishing sensor information by JSON converting object to a string
    publish.single("topic/sensorTemperature", json.dumps(dht22), hostname = "test.mosquitto.org")
    publish.single("topic/sensorTemperature", json.dumps(arduino), hostname = "test.mosquitto.org")

    # Printing JSON objects
    print(dht22)
    print(arduino)
    time.sleep(2)

if __name__ == '__main__':

  try:
    main()
  except KeyboardInterrupt:
    pass
  finally:

    GPIO.cleanup()

以下是从我的笔记本电脑订阅和接收数据的代码:

import paho.mqtt.client as mqtt
import json

# This is the Subscriber
def on_connect(client, userdata, flags, rc):
    print("Connected with result code " + str(rc))
    client.subscribe("topic/sensorTemperature")


def on_message(client, userdata, msg):
    print(json.loads(msg.payload)) #converting the string back to a JSON object

client = mqtt.Client()

client.on_connect = on_connect
client.on_message = on_message

client.connect("test.mosquitto.org", 1883, 60)

client.loop_forever()

我想要做的是现在从我的笔记本电脑发布一些内容(可能与订阅者在相同的代码中,或者在一个单独的文件中,只发布一条消息到同一主题 - "topic/sensorTemperature")。但我的问题是:我如何在我的Raspberry Pi上发布和订阅消息(在我发布的第一个代码中)?由于我在无限循环中向我的笔记本电脑发布消息,因此我还需要一个无限循环来订阅相同(或不同的主题)来接收消息。你如何一次运行其中两个循环?我需要两个不同的线程吗?

谢谢。

3 个答案:

答案 0 :(得分:1)

最简单的方法是在Raspberry上并行启动另一个Python进程(类似于您的笔记本电脑的脚本),处理从笔记本电脑收到的消息。

但是如果你想在一个脚本中实现所有东西,你可以扩展你的第二个代码片段(处理消息),实现第一个片段(发布传感器数据)。

当然,在这种情况下,您无法使用 loop_forever()。当你调用loop_forever()时,它将永远不会返回,直到客户端调用disconnect(),因此你不能处理收到的消息(主线程被阻止)。 Paho客户端还具有例程 loop() loop_start()/ loop_stop()来控制网络循环。 看看他们:

1)函数 loop()可以将超时作为参数。它将阻止,直到新消息到达或时间结束。在第一种情况下 - 预先处理收到的消息并计算下次发布之前的时间。将此时间作为参数再次传递给loop()。在第二种情况下,只需发布​​数据并调用循环(),直到下次发布(在您的示例中为2秒)。

2) loop_start()/ loop_stop()启动和停止后台线程,为您发送和接收(和处理)数据。创建客户端,注册 on_message()回调,连接/订阅,并调用loop_start()来启动此线程。主线程现在可以免费使用 - 使用第一个片段的逻辑(2秒睡眠循环)。

答案 1 :(得分:1)

根据Sergey的建议,您可以使用 loop_start 创建一个单独的线程来接收消息。

以下是您的主要功能:

def main():
    # Create a new client for receiving messages
    client = mqtt.Client()
    client.on_connect = on_connect
    client.on_message = on_message
    client.subscribe(topic)
    client.connect(mqttserver)
    client.loop_start()
    while True:
        #code for publishing
        pass

答案 2 :(得分:1)

只需在subscribing script之前将代码从publishing script放入while True:,然后将loop_forever()替换为loop_start()。当脚本在loop_stop()之前退出时使用GPIO.cleanup()