如何使用 java 客户端并发发布 MQTT 消息?
how to publish MQTT messages concurrently using java client?
我正在尝试使用 5 个 java 客户端并发发布 MQTT 消息,这样每个 java 客户端就特定主题同时向 MQTT 代理 (HIVEMQ) 发布 1000 条消息
我打开了多个线程,每个线程创建一个 mqtt 客户端并使用 ssl 连接到代理并尝试同时发布 1000 条消息,消息正在发送但所有连接都没有成功到代理,我继续异常
Client is not connected (32104)
at org.eclipse.paho.client.mqttv3.internal.ExceptionHelper.createMqttException(ExceptionHelper.java:31)
at org.eclipse.paho.client.mqttv3.internal.ClientComms.sendNoWait(ClientComms.java:199)
at org.eclipse.paho.client.mqttv3.MqttAsyncClient.publish(MqttAsyncClient.java:1355)
at org.eclipse.paho.client.mqttv3.MqttClient.publish(MqttClient.java:583)
at org.eclipse.paho.client.mqttv3.MqttClient.publish(MqttClient.java:575)
at com.test.MqttPublishSample.publishMessages(MqttPublishSample.java:122)
at com.test.MqttPublishSample.lambda$start[=11=](MqttPublishSample.java:74)
at java.base/java.lang.Thread.run(Thread.java:834)
public class MqttPublishSample {
public static void main(String... args) throws InterruptedException {
new MqttPublishSample().start();
}
public void start() throws InterruptedException {
for(int i=0;i<5;i++){
new Thread(()->{
MqttClient client = null;
try {
client = obtainConnection();//code to obtain connection using MqttClient
publishMessages(client);//code to publish message using simple for loop
} catch (MqttException e) {
e.printStackTrace();
}
}).start();
}
}
public MqttClient obtainConnection() throws MqttException {
String clientId = "sslTestClient"+ThreadLocalRandom.current().nextInt(0,5);
MqttClient client = null;
try {
client = new MqttClient("ssl://localhost:8883", clientId, new MemoryPersistence());
} catch (MqttException e) {
e.printStackTrace();
}
MqttConnectOptions mqttConnectOptions = new MqttConnectOptions();
mqttConnectOptions.setUserName("user1");
mqttConnectOptions.setPassword("pass1".toCharArray());
try {
mqttConnectOptions.setSocketFactory(getTruststoreFactory());
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("connecting...");
client.connect(mqttConnectOptions);
return client;
}
我希望所有客户端都能成功连接到代理并毫无例外地发布消息
可能是因为您在线程上使用了相同的 clientID,因此,服务器将断开重复连接。当您使用 LocalThreadRandom 时,有可能发生碰撞(足够大,因为只有 5 个选择)。您可以使用 generateClientId() 提供的唯一标识符或在线程之间共享一个方法来跟踪它们。
我正在尝试使用 5 个 java 客户端并发发布 MQTT 消息,这样每个 java 客户端就特定主题同时向 MQTT 代理 (HIVEMQ) 发布 1000 条消息
我打开了多个线程,每个线程创建一个 mqtt 客户端并使用 ssl 连接到代理并尝试同时发布 1000 条消息,消息正在发送但所有连接都没有成功到代理,我继续异常
Client is not connected (32104)
at org.eclipse.paho.client.mqttv3.internal.ExceptionHelper.createMqttException(ExceptionHelper.java:31)
at org.eclipse.paho.client.mqttv3.internal.ClientComms.sendNoWait(ClientComms.java:199)
at org.eclipse.paho.client.mqttv3.MqttAsyncClient.publish(MqttAsyncClient.java:1355)
at org.eclipse.paho.client.mqttv3.MqttClient.publish(MqttClient.java:583)
at org.eclipse.paho.client.mqttv3.MqttClient.publish(MqttClient.java:575)
at com.test.MqttPublishSample.publishMessages(MqttPublishSample.java:122)
at com.test.MqttPublishSample.lambda$start[=11=](MqttPublishSample.java:74)
at java.base/java.lang.Thread.run(Thread.java:834)
public class MqttPublishSample {
public static void main(String... args) throws InterruptedException {
new MqttPublishSample().start();
}
public void start() throws InterruptedException {
for(int i=0;i<5;i++){
new Thread(()->{
MqttClient client = null;
try {
client = obtainConnection();//code to obtain connection using MqttClient
publishMessages(client);//code to publish message using simple for loop
} catch (MqttException e) {
e.printStackTrace();
}
}).start();
}
}
public MqttClient obtainConnection() throws MqttException {
String clientId = "sslTestClient"+ThreadLocalRandom.current().nextInt(0,5);
MqttClient client = null;
try {
client = new MqttClient("ssl://localhost:8883", clientId, new MemoryPersistence());
} catch (MqttException e) {
e.printStackTrace();
}
MqttConnectOptions mqttConnectOptions = new MqttConnectOptions();
mqttConnectOptions.setUserName("user1");
mqttConnectOptions.setPassword("pass1".toCharArray());
try {
mqttConnectOptions.setSocketFactory(getTruststoreFactory());
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("connecting...");
client.connect(mqttConnectOptions);
return client;
}
我希望所有客户端都能成功连接到代理并毫无例外地发布消息
可能是因为您在线程上使用了相同的 clientID,因此,服务器将断开重复连接。当您使用 LocalThreadRandom 时,有可能发生碰撞(足够大,因为只有 5 个选择)。您可以使用 generateClientId() 提供的唯一标识符或在线程之间共享一个方法来跟踪它们。