Spring STOMP-用户的目的地

应用程序可以发送针对特定用户的消息,并且Spring的STOMP支持识别以/user/为前缀的destination。例如,客户端可能会订阅/user/queue/position-updates的destination。UserDestinationMessageHandler处理此destination,并将其转换为特定于用户会话的destination(例如/queue/position-updates-user123)。这提供了订阅通用命名destination的便利,同时确保与其他订阅相同目的地的用户没有冲突,以便每个用户都可以接收到独特的股票位置更新。

使用用户目标时,配置代理和应用程序目标前缀非常重要,如启用 STOMP 中所示,否则代理将处理仅应由 UserDestinationMessageHandler 处理的“/user”前缀消息。

在发送端,消息可以发送到诸如 /user/{username}/queue/position-updates 之类的destination,而该destination又由 UserDestinationMessageHandler 转换为一个或多个destination,每个destination对应与用户关联的每个会话。这使得应用程序中的任何组件都可以发送针对特定用户的消息,而无需知道除用户名和通用destination之外的任何信息。这也可以通过注释和消息模板来支持。

消息处理方法可以通过@SendToUser注解向正在处理的消息关联的用户发送消息(也可以在类级别上支持,以共享一个共同的目的地),如下例所示:

@Controller
public class PortfolioController {

	@MessageMapping("/trade")
	@SendToUser("/queue/position-updates")
	public TradeResult executeTrade(Trade trade, Principal principal) {
		*// ...*return tradeResult;
	}
}

如果用户有多个会话,则默认情况下,订阅给定destination的所有会话都会成为目标。然而,有时,可能需要仅针对发送正在处理的消息的会话。你可以通过将broadcast属性设置为 false 来实现此目的,如以下示例所示:

@Controller
public class MyController {

	@MessageMapping("/action")
	public void handleAction() throws Exception{
		*// raise MyBusinessException here*
	}

	@MessageExceptionHandler
	@SendToUser(destinations="/queue/errors", broadcast=false)
	public ApplicationError handleException(MyBusinessException exception) {
		*// ...*return appError;
	}
}

虽然用户destination通常意味着已认证的用户,但这并不是严格要求的。未与已认证用户关联的WebSocket会话可以订阅用户destination。在这种情况下,@SendToUser注解的行为与broadcast=false完全相同(即,仅定位发送正在处理的消息的会话)。

你可以通过注入由Java配置或XML命名空间创建的SimpMessagingTemplate来向用户destinations发送消息。(如果需要通过@Qualifier进行限定,bean名称为brokerMessagingTemplate)。以下示例展示了如何执行此操作:

@Service
public class TradeServiceImpl implements TradeService {

	private final SimpMessagingTemplate messagingTemplate;

	@Autowired
	public TradeServiceImpl(SimpMessagingTemplate messagingTemplate) {
		this.messagingTemplate = messagingTemplate;
	}

	*// ...*public void afterTradeExecuted(Trade trade) {
		this.messagingTemplate.convertAndSendToUser(
				trade.getUserName(), "/queue/position-updates", trade.getResult());
	}
}

当你将用户目标与外部消息代理一起使用时,应检查代理文档了解如何管理不活动的队列,以便在用户会话结束后,删除所有唯一的用户队列。例如,当你使用 /exchange/amq.direct/position-updates 等destinations时,RabbitMQ 会创建自动删除队列。因此,在那种情况下,客户端可以订阅/user/exchange/amq.direct/position-updates。同样,ActiveMQ有配置选项来清除不活动的目的地。

在多应用程序服务器场景中,由于用户连接到不同的服务器,因此用户destination可能仍未解析。在这种情况下,你可以配置一个destination来广播未解析的消息,以便其他服务器有机会尝试。这可以通过 Java 配置中 MessageBrokerRegistryuserDestinationBroadcast 属性和 XML 中 message-broker 元素的 user-destination-broadcast 属性来完成。

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

李昂的数字之旅

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值