Spring 与使用STOMP消息

本文介绍如何使用STOMP在WebSocket上建立消息传递系统,包括配置STOMP代理中继、处理客户端消息、发送消息到客户端以及为目标用户发送消息。还讨论了处理消息异常的方法。
一、STOMP

像HTTP在TCP套接字之上添加了请求-响应模型层一样,STOMP在WebSocket之上提供了一个基于帧的线路格式(frame-based wireformat)层,用来定义消息的语义。

1.1 启用STOMP消息功能

在这里插入图片描述
WebSocketStompConfig使用了@EnableWebSocketMessageBroker注解。这表明这个配置类不仅配置了WebSocket,还配置了基于代理的STOMP消息。它重载了registerStompEndpoints()方法,将“/marcopolo”注册为STOMP端点。这个路径与之前发送和接收消息的目的地路径有所不同。这是一个端点,客户端在订阅或发布消息到目的地路径前,要连接该端点。

WebSocketStompConfig还通过重载configureMessageBroker()方法配置了一个简单的消息代理。这个方法是可选的,如果不重载它的话,将会自动配置一个简单的内存消息代理,用它来处理以“/topic”为前缀的消息。但是在本例中,我们重载了这个方法,所以消息代理将会处理前缀为“/topic”和“/queue”的消息。除此之外,发往应用程序的消息将会带有“/app”前缀。
在这里插入图片描述

启用STOMP代理中继

在这里插入图片描述
上述configureMessageBroker()方法的第一行代码启用了STOMP代理中继(broker relay)功能,并将其目的地前缀设置为“/topic”和“/queue”。这样的话,Spring就能知道所有目的地前缀为“/topic”或“/queue”的消息都会发送到STOMP代理中。根据你所选择的STOMP代理不同,目的地的可选前缀也会有所限制。
在这里插入图片描述

enableStompBrokerRelay()和setApplicationDestinationPrefixes()方法都接收可变长度的String参数,所以我们可以配置多个目的地和应用前缀。例如:
在这里插入图片描述
默认情况下,STOMP代理中继会假设代理监听localhost的61613端口,并且客户端的username和password均为“guest”。如果你的STOMP代理位于其他的服务器上,或者配置成了不同的客户端凭证,那么我们可以在启用STOMP代理中继的时候,需要配置这些细节信息:
在这里插入图片描述

1.2 处理来自客户端的STOMP消息

在这里插入图片描述

处理订阅

除了@MessagingMapping注解以外,Spring还提供了@SubscribeMapping注解。与@MessagingMapping注解方法类似,当收到STOMP订阅消息的时候,带有@SubscribeMapping注解的方法将会触发。

编写JavaScript客户端

如下的程序清单展现了一些JavaScript客户端代码,它会连接“/marcopolo”端点并发送“Marco!”消息。
在这里插入图片描述

1.3 发送消息到客户端

Spring提供了两种发送数据给客户端的方法:

  • 作为处理消息或处理订阅的附带结果;
  • 使用消息模板。

在处理消息之后,发送消息

如果你想要在接收消息的时候,同时在响应中发送一条消息,那么需要做的仅仅是将内容返回就可以了,方法签名不再是使用void。例如,如果你想发送“Polo!”消息作为“Marco!”消息的回应,那么只需将handleShout()修改为如下所示:
在这里插入图片描述

我们可以通过为方法添加@SendTo注解,重载目的地:
在这里插入图片描述
按照这个@SendTo注解,消息将会发布到“/topic/shout”。所有订阅这个主题的应用(如客户端)都会收到这条消息。
这样的话,handleShout()在收到一条消息的时候,作为响应也会发送一条消息。按照类似的方式,@SubscribeMapping注解标注的方式也能发送一条消息,作为订阅的回应。例如,通过为控制器添加如下的方法,当客户端订阅的时候,将会发送一条Shout信息:
在这里插入图片描述
这里的@SubscribeMapping注解表明当客户端订阅“/app/marco”(“/app”是应用目的地的前缀)目的地的时候,将会调用handleSubscription()方法。它所返回的Shout对象将会进行转换并发送回客户端。

在应用的任意地方发送消息

@MessageMapping和@SubscribeMapping提供了一种很简单的方式来发送消息,这是接收消息或处理订阅的附带结果。不过,Spring的SimpMessagingTemplate能够在应用的任何地方发送消息,甚至不必以首先接收一条消息作为前提。

使用SimpMessagingTemplate的最简单方式是将它(或者其接口SimpMessageSendingOperations)自动装配到所需的对象中。
在客户端需要添加如下代码:
在这里插入图片描述
在这里插入图片描述

在服务器端,我们可以使用SimpMessagingTemplate将所有新创建的Spittle以消息的形式发布到“/topic/spittlefeed”主题上。如下程序清单展现的SpittleFeedServiceImpl就是实现该功能的简单服务:
在这里插入图片描述

二、为目标用户发送消息

我们可以使用SpringSecurity来认证用户,并为目标用户处理消息。在使用Spring和STOMP消息功能的时候,我们有三种方式利用认证用户:

  • @MessageMapping和@SubscribeMapping标注的方法能够使用Principal来获取认证用户;
  • @MessageMapping、@SubscribeMapping和
  • @MessageException方法返回的值能够以消息的形式发送给认证用户;
  • SimpMessagingTemplate能够发送消息给特定用户。

2.1 在控制器中处理用户的消息

在这里插入图片描述
在这里插入图片描述

2.2 为指定用户发送消息

在这里插入图片描述
在broadcastSpittle()中,如果给定Spittle对象的消息中包含了类似于用户名的内容(也就是以“@”开头的文本),那么一个新的Notification将会发送到名为“/queue/notifications”的目的地上。因此,如果Spittle中包含“@jbauer”的话,Notification将会发送到“/user/jbauer/queue/notifications”目的地上。

三、处理消息异常

在Spring MVC中,如果在请求处理中,出现异常的话,@ExceptionHandler方法将有机会处理异常。与之类似,我们也可以在某个控制器方法上添加@MessageExceptionHandler注解,让它来处理@MessageMapping方法所抛出的异常。
例如,考虑如下的方法,它会处理消息方法所抛出的异常:
在这里插入图片描述
按照最简单的形式,@MessageExceptionHandler标注的方法能够处理消息方法中所抛出的异常。但是,我们也可以以参数的形式声明它所能处理的异常:
在这里插入图片描述
或者,以数组参数的形式指定多个异常类型:
在这里插入图片描述
尽管它只是以日志的方式记录了所发生的错误,但是这个方法可以做更多的事情。例如,它可以回应一个错误:
在这里插入图片描述
在这里,如果抛出SpittleException的话,将会记录这个异常,然后将其返回。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

书香水墨

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

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

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

打赏作者

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

抵扣说明:

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

余额充值