由于客户端订阅了“/user/monitor”队列,服务器的@SubscribeMapping注解必须是“/user/monitor”而不是“/user/monitor{something}”。
服务器可以根据您的身份验证方案了解客户端是什么。如果您使用 sockjs websocket,您可以使用 HTTP 身份验证,从而利用 Spring Security,并在您的函数中添加一个“Principal”参数来保存用户信息:
@SubscribeMapping("/monitor")
public void init(Principal p) {
String user = p.getName();
messagingTemplate.convertAndSendToUser(user, "/monitor", getOps());
}
如果您不使用 http 身份验证,您可以向服务器发送用户信息,例如添加一个自定义 STOMP 标头,该标头可以使用 SimpMessageHeaderAccessor 从服务器访问:
@SubscribeMapping("/monitor")
public void init(SimpMessageHeaderAccessor accessor) {
String user = accessor.getFirstNativeHeader("userid");
messagingTemplate.convertAndSendToUser(user, "/monitor", getOps());
}
另一种方法可能是订阅包含用户信息的不同队列名称(这可能是您的建议)。客户端必须使用类似于以下的代码:
stompClient.subscribe('/topic/byUser/'+userId, function(msg) {
console.log(msg);
});
以便服务器可以通过这种方式访问它:
@SubscribeMapping("/byUser/{userId}")
public void init(@DestinationVariable String userId) {
messagingTemplate.convertAndSendToUser(userId, "/monitor", getOps());
}
但在这种情况下,请记住该队列是公共的,因此如果其他客户端知道其他客户端名称,则可以访问他们的消息。