【发布时间】:2016-03-11 07:52:49
【问题描述】:
我的目标是从前端向后端发送一个请求并接收多个响应。我使用 WebSocket 是因为响应非常频繁,而 WebSocket 似乎是最好的协议,并且 SseEmitter 从后端发送多个响应。
这是我的请求控制器:
@MessageMapping("/emitter")
@SendTo("/topic/response")
public SseEmitter output(RunData runData) throws Exception {
SseEmitter emitter = new SseEmitter();
new Thread(new Runnable() {
@Override
public void run() {
try {
RemoteHostController rhc = new RemoteHostController(runData);
rhc.execute();
while (rhc.getActiveCount() > 0) {
emitter.send(rhc.getAllOutput());
Thread.sleep(2000);
}
emitter.complete();
} catch (Exception ee) {
ee.printStackTrace();
emitter.completeWithError(ee);
}
}
}).start();
return emitter;
}
RemoteHostController 正在管理连接,getAllOutput 从主机返回输出。
前端应用程序正在运行非常简单的 index.html,它使用 Stomp 和 SockJS 连接到 websocket,将数据发送到服务器并生成
带有来自响应的数据的标签:
function connect() {
var socket = new SockJS('http://localhost:8080/emitter');
stompClient = Stomp.over(socket);
stompClient.connect({}, function(frame) {
setConnected(true);
console.log('Connected: ' + frame);
stompClient.subscribe('/topic/response', function(greeting){
showOutput(greeting.body);
});
});
}
function sendData() {
var hostname = document.getElementById('hostname').value;
var username = document.getElementById('username').value;
var password = document.getElementById('password').value;
var command = document.getElementById('command').value;
stompClient.send("/app/emitter", {}, JSON.stringify({ 'hostname': hostname,
'username': username,
'password': password,
'command': command}));
}
function showOutput(message) {
var response = document.getElementById('response');
var p = document.createElement('p');
p.style.wordWrap = 'break-word';
p.appendChild(document.createTextNode(message));
response.appendChild(p);
}
当我将数据发送到后端时,我得到的唯一响应是:
{"timeout":null}
这是 SseEmitter 超时字段,当我更改超时时,它将返回 {"timeout":<timeout_value>}。
我可以在日志中看到 RemoteHostController 正在连接到主机并正确执行命令。
我做错了吗?还是WebSocket只支持一请求一响应通信?
【问题讨论】:
-
您看到超时字段的原因是因为您的控制器方法正在返回一个 SseEmitter 并且这就是发送到 SendTo 目的地的内容。 Websockets 可以异步发送,因此没有实际的请求/响应。我对 SSE 不熟悉,但您似乎需要在客户端使用 EventSource(请参阅 coderanch.com/t/658238/Spring/… 作为示例。我在我们编写的应用程序中使用了 WebSockets,您使用 SimpMessagingTemplate 在后端发送内容。如果时间允许,我可能会发布解决方案
-
我有一个
SseEmitter的示例,但它是通过EventSource将流数据发送到此Project 中的客户端。您可以在那里查看 SseEmitter 代码。由于 Internet Explorer 不支持EventSource,因此我稍后将移至WebSocket。另外我认为我不需要SseEmitter与WebSocket一起使用,我认为它通过@Rob Baily 指出的EventSource发送数据
标签: java spring websocket spring-boot stomp