MQTT断线重连及订阅消息恢复

版权声明:版权归“行知致简”所有,转载请保留 https://blog.csdn.net/paladinzh/article/details/88053530

MQTT断线重连及订阅消息恢复

工具

MQTT客户端是用的 paho http://www.eclipse.org/paho/

问题

采用以下配置

        connOpts = new MqttConnectOptions(); 
        connOpts.setCleanSession(true);
        connOpts.setConnectionTimeout(10);
        connOpts.setKeepAliveInterval(90);
        connOpts.setAutomaticReconnect(true);

网络断开后,客户端会进行重连,但是重连之前订阅的主题就失效了,不再接受之前订阅主题的消息。

解决方法

因为配置里将cleanSession 设为 true ,当客户端掉线时 ,服务器端会清除 客户端 session 。 重连后 客户端会有一个新的session。

第一种

把配置里的 cleanSession 设为false,客户端掉线后 服务器端不会清除session,当重连后可以接收之前订阅主题的消息。当客户端上线后会接受到它离线的这段时间的消息

但是这个只是进行了重连,重连后还需要再次发起订阅

对于离线消息为什么不能接收到,我想说的是 你很有可能在再次连接的时候又重新订阅了该topic。这样是接收不到离线消息的。
离线消息是订阅了某个topic的client 在断开连接以后,在接下来又重新连接的时候,任然可以收到在它断开连接的这段时间内,该topic上的消息。这就需要将client的 setCleanSession 设置为false,这样服务器才能保留它的session,再次建立连接的时候,它就会继续使用这个session了。 

注意:clientId 是不能更改的。

第二种

通过查看 connect 方法的代码发现

这里写图片描述

如果我们设置的 callback实现了 MqttCallbackExtended接口,就会把它设置给connectActionListener,我再来看看MqttCallbackExtended接口。 
这里写图片描述

可以看到这个接口拓展了一个方法,该方法会在连接到服务器成功时调用,正符合我们的要求。我只需要实现这个接口并在这个方法里从新订阅之前的主题就行了。

这种方法重连之后不会收到离线期间该主题的消息

第三种:

//设置断开后重新连接 
options.setAutomaticReconnect(true);

但这样重连是实现了,但是之前订阅的主题却接收不到消息了,需要重新订阅主题才能正常接收消息,那我这个重新订阅的代码要怎么再放进去呢,反正不是再connectionLost里就是了,那是后连接还没有重连连上!

继续看MQTT的connec的源码发现了一段代码使我找到了解决方案

MqttReconnectCallback 是实现MqttCallbackExtended接口的

发现comms中有设置重连的回调对象,但是怎么把这个回调由我们来主动放进去呢?继续往下看源码可以发现

也就是如果我们在之前放入client的回调对象是实现的 MqttCallbackExtended 接口,则MQTT会将我们的回调对象放入 connectActionListener 中 然后由 connectActionListener实现具体的connect

接下来我们不callback 对象改为实现 MqttCallbackExtended这个接口,然后实现下面方法,

@Override
public void connectComplete(boolean reconnect, String serverURI) {
		//连接成功后调用
      client.subscribe(topics,Qos);//具体订阅代码
}

就可以解决MQTT重连后无法订阅的问题。

猜你喜欢

转载自blog.csdn.net/paladinzh/article/details/88053530
今日推荐