【MQTT篇 心跳机制】

在医院里,医生利用心跳来判断患者是否还有生命体征。对于MQTT服务器来说,它要判断一台MQTT客户端是否依然保持连接可以检查这台客户端是不是经常发送消息给服务端。如果服务端经常收到客户端的消息,那么没问题,这个客户端肯定在线。

但是有些客户端并不经常发送消息给服务端。对于这种客户端,服务端可以使用类似心跳检测的方法,来判断客户端是否在线。

1

不过客户端设备没有心脏,自然不会跳动。所以MQTT协议为它们配上一个类似心脏的机制,这个心脏机制就是让客户端在没有向服务端发送信息时,可以定时向服务端发送一条消息。这条用于心跳机制的消息也被称作心跳请求(PINGREQ)。
2
心跳请求的作用正是用于告知服务端,当前客户端依然在线。服务端在收到客户端的心跳请求后,会回复一条消息。这条回复消息被称作心跳响应(PINGRESP)。

由于心跳请求是客户端定时发送的,一旦服务端发现客户端停止发送请求信息,那么服务端就会知道,这台客户端已经断开了连接。

这个心跳机制不仅可以用于服务端判断客户端是否保持连接,也可以用于客户端判断自己与服务端是否保持连接。如果客户端在发送心跳请求(PINGREQ)后,没有收到服务端的心跳响应(PINGRESP),那么客户端就会认为自己与服务端的连接已经被断开了。

以上是心跳机制(Keep Alive)的简单介绍,为了更深入的了解心跳机制,我们来看一下一下MQTT客户端在连接服务端的过程。请见下图,客户端连接服务端时会像服务端发送CONNECT报文。
4
我刚刚给大家讲过,在心跳机制中,客户端要定时向服务端发送心跳请求(PINGREQ)报文。那么客户端发送心跳请求的时间间隔是多少呢?

这个心跳时间间隔是我们在开发客户端时进行设置的。假如我们使用ESP8266开发板作为物联网客户端,那么我们在编写控制程序时,会在程序中对心跳时间间隔进行设置。

设置好心跳时间间隔后,客户端就知道多久要发送一条心跳请求给服务端。但是这里存在一个问题。光是客户端知道心跳时间间隔还不够,服务端也需要知道客户端的心跳时间间隔,这样服务端才能定时检查客户端的心跳请求消息。

因此,在客户端连接服务端时,会将心跳时间间隔信息放入CONNECT报文。也就是上图中最后一行的信息keepAlive。这个keepAlive正是用于告知服务端心跳时间间隔的。

以上示例图中我们看到keepAlive数值为60。这就意味着,客户端的心跳间隔时间是60秒。

接下来我要给您讲解的内容十分关键,请您务必留意。

在继续后面的学习以前我们先来问您个问题。假如客户端的心跳间隔时间是60秒,那么服务端是不是每隔60秒就检查一次客户端是否发来心跳请求呢?

我们本节课刚开始的时候曾给大家介绍过,如果客户端在心跳时间间隔内发布了消息给服务端,那么服务端不需要客户端发送心跳请求也可以确定该客户端肯定在线。
3
但是当客户端在心跳间隔内没有发布消息给服务端,这时客户端会主动发送一个心跳请求消息给服务端。以表明自己仍让在线

6
简而言之,客户端在心跳间隔时间内,如果有消息发布,那就不发布心跳请求。但是在心跳间隔时间内,客户端没有消息发布,那么它就会发布一条心跳请求给服务端,这个心跳请求的目的就是为了告诉服务端,“我还在线,你放心吧。”

另外,在实际运行中,如果服务端没有在1.5倍心跳时间间隔内收到客户端发布消息(PUBLISH)或发来心跳请求(PINGREQ),那么服务端就会认为这个客户端已经掉线。

举例来说,如果心跳时间间隔是60秒。那么服务端在90秒内没有收到客户端发布的消息也没有收到PINGREQ请求,那么它就会认为客户端已经掉线。

另外,心跳机制不仅仅用于服务端判断客户端是否在线。客户端也可以利用这一机制来判断自己是否与服务端仍保持连接。如果客户端发送了心跳请求(PINGREQ)给服务端一段时间后,仍然没有收到服务端回复的心跳确认。那么客户端也会认为自己已经断开了与服务端的连接。

了解了MQTT心跳机制后,不知道您有没有想过,如果服务端知道了某一台客户端已经掉线,它会采取什么措施吗?关于这个问题答案,我们将在下一节课为您讲解。

  • 2
    点赞
  • 30
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
要实现 MQTT Java 客户端的心跳检测,你可以使用 Eclipse Paho 提供的 MQTT Java 客户端库。以下是一个简单的示例代码: ```java import org.eclipse.paho.client.mqttv3.*; public class MqttHeartbeat { public static void main(String[] args) throws MqttException { String broker = "tcp://mqtt.example.com:1883"; String clientId = "heartbeatClient"; String topic = "heartbeat"; MqttClient client = new MqttClient(broker, clientId); MqttConnectOptions options = new MqttConnectOptions(); options.setCleanSession(true); client.connect(options); // 定义心跳消息 MqttMessage heartbeatMessage = new MqttMessage("heartbeat".getBytes()); heartbeatMessage.setQos(0); heartbeatMessage.setRetained(false); // 设置心跳定时器 int interval = 5000; // 心跳间隔,单位为毫秒 MqttPingSender pingSender = new MqttPingSender(client, topic, heartbeatMessage, interval); client.setPingSender(pingSender); // 订阅心跳回复消息 client.subscribe(topic); // 发布心跳消息 client.publish(topic, heartbeatMessage); // 等待心跳回复消息 client.setCallback(new MqttCallback() { @Override public void messageArrived(String topic, MqttMessage message) throws Exception { System.out.println("Received heartbeat response: " + message.toString()); } @Override public void connectionLost(Throwable cause) { System.out.println("Connection lost: " + cause.getMessage()); } @Override public void deliveryComplete(IMqttDeliveryToken token) { } }); // 保持程序运行 try { Thread.sleep(Long.MAX_VALUE); } catch (InterruptedException e) { e.printStackTrace(); } // 断开连接 client.disconnect(); } } class MqttPingSender implements MqttPingSender { private MqttClient client; private String topic; private MqttMessage message; private int interval; private MqttTopic mqttTopic; public MqttPingSender(MqttClient client, String topic, MqttMessage message, int interval) { this.client = client; this.topic = topic; this.message = message; this.interval = interval; this.mqttTopic = client.getTopic(topic); } @Override public void init() throws MqttException { } @Override public void schedule(long delayInMilliseconds) throws MqttException { try { Thread.sleep(delayInMilliseconds); } catch (InterruptedException e) { e.printStackTrace(); } } @Override public void start() throws MqttException { schedule(interval); sendHeartbeat(); } @Override public void stop() throws MqttException { } private void sendHeartbeat() throws MqttException { mqttTopic.publish(message); schedule(interval); } } ``` 在上述示例中,我们使用 Eclipse Paho 提供的 `MqttClient` 类来创建 MQTT 客户端,并设置连接选项。然后,我们定义了心跳消息,并通过设置心跳定时器来定期发布心跳消息。同时,我们还订阅了心跳回复消息,并在回调函数中处理接收到的回复消息。 你可以根据自己的需求修改示例代码中的 broker、clientId、topic、心跳间隔等参数来适应你的实际情况。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

暮誠雪

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值