Я пытаюсь реализовать веб-сокеты, используя spring с java/веб-приложением, чтобы позволить ему обмениваться сообщениями с приложением, написанным на С++, с использованием qt (и библиотеки веб-сокетов из него).
У меня есть в моем приложении java/spring эта конфигурация:
WebScoketConfig.java
@Configuration
@EnableWebSocket
public class WebSocketConfig implements WebSocketConfigurer {
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
registry.addHandler(new SocketHandler(), "/name");
}
}
SocketHandler.java
@Component
public class SocketHandler extends TextWebSocketHandler {
List<WebSocketSession> sessions = new CopyOnWriteArrayList<>();
@Override
public void handleTextMessage(WebSocketSession session, TextMessage message) throws InterruptedException, IOException {
Map<String, String> value = new Gson().fromJson(message.getPayload(), Map.class);
session.sendMessage(new TextMessage("Hello " + value.get("name") + " !"));
}
@Override
public void afterConnectionEstablished(WebSocketSession session) throws Exception {
sessions.add(session);
}
}
и я создал очень простой проект qt-creator с функцией main
и одним классом MainWindow
с двумя объектами: lineEdit
, где пользователь вводит сообщение для отправки на сервер, и pushButton
, чтобы продолжить отправку данных.
В моем классе MainWindow
я реализую этот слот для обработки обмена данными:
void MainWindow::on_pushButton_clicked()
{
QString message = this->ui->lineEdit->text();
QWebSocket m_webSocket;
m_webSocket.open(QUrl(QStringLiteral("ws://localhost:8080/name")));
m_webSocket.sendTextMessage("Hello " + message + " !");
m_webSocket.close();
}
Но когда я запускаю оба приложения и пытаюсь отправить сообщение для java/веб-приложения, ничего не происходит. Я почти уверен, что ошибка, которую я совершил, связана с c++/qt, поскольку на стороне java/spring у меня есть код html/javascript, который позволяет мне тестировать обмен сообщениями и работает нормально.
Кто-нибудь может сказать мне, что я делаю неправильно здесь?
обновление: минимальный воспроизводимый пример — java/spring
проект может быть сгенерирован с помощью start.spring.io, только с зависимостью spring-websocket. помимо двух файлов, которые я уже добавил выше, в проекте будет:
ресурсы/статические/index.html
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Hello WebSocket</title>
<link href="/main.css" rel="stylesheet">
</head>
<body>
<table>
<tr>
<td>
<button id="connect" type="button" onclick="connect();">Connect</button>
<button id="disconnect" type="button" disabled="disabled" onclick="disconnect();">Disconnect</button>
</td>
<td>
<label for="name">What is your name?</label>
<input type="text" id="name" placeholder="Your name here...">
<button id="send" type="button" onclick="send();">Send</button>
</td>
</tr>
</table>
<hr>
<table id="conversation" border="2">
<thead>
<tr>
<th>Greetings</th>
</tr>
</thead>
<tbody id="greetings">
</tbody>
</table>
<script src="/app.js"></script>
</body>
</html>
ресурсы/app.js
вар ВС;
function connect() {
ws = new WebSocket('ws://localhost:8080/name');
ws.onmessage = function(text) {
var tr = document.createElement("tr");
var td = document.createElement("td");
td.innerText = text.data;
tr.appendChild(td);
document.querySelector("#greetings").appendChild(tr);
}
document.querySelector("#connect").setAttribute("disabled", "disabled");
document.querySelector("#disconnect").removeAttribute("disabled");
document.querySelector("#conversation").style.display = 'block';
document.querySelector("#greetings").innerHTML = "";
}
function disconnect() {
if (ws != null)
ws.close();
document.querySelector("#connect").removeAttribute("disabled");
document.querySelector("#disconnect").setAttribute("disabled", "disabled");
document.querySelector("#conversation").style.display = 'none';
document.querySelector("#greetings").innerHTML = "";
}
function send() {
var name = document.querySelector("#name");
var data = JSON.stringify({'name': name.value});
ws.send(data);
}
после сборки с mvn package
просто запустите с java -jar target/app.jar
.
обновление: минимальный воспроизводимый пример — c++/qt
проект создается с помощью qt-creator, как тип qt-widget. Он создаст проект с 5 файлами: websocket.pro
, mainwindow.ui
, mainwindow.h
, mainwindow.cpp
и main.cpp
.
Откройте mainwindow.ui
и добавьте lineEdit
и pushButton
с панели инструментов. щелкните правой кнопкой мыши кнопку, выберите Go to slot
и выберите clicked()
. Добавьте код выше.
Обновление 2
void MainWindow::on_pushButton_clicked()
{
QString message = ui->lineEdit->text();
connect(&m_webSocket, &QWebSocket::connected, [this, message](){
QJsonObject object
{
{"name", message}
};
QJsonDocument d(object);
m_webSocket.sendTextMessage(d.toJson().toStdString().c_str());
m_webSocket.close();
});
m_webSocket.open(QUrl(QStringLiteral("ws://localhost:8080/name")));
}
QWebSocket::sendTextMessage
на самом деле не отправляет сообщение немедленно. Скорее он откладывает отправку до тех пор, пока вы не вернете управление циклу обработки событий Qt. В этом случае сообщение может быть потеряно, когда вашQWebSocket
выходит за рамки. - person G.M.   schedule 09.04.2020QWebSocket
m_webSocket
членом данных классаMainWindow
. - person G.M.   schedule 10.04.2020m_websocket
члена данныхMainWindow
не делает эту работу. - person Kleber Mota   schedule 10.04.2020