一、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的话,将会记录这个异常,然后将其返回。
本文介绍如何使用STOMP在WebSocket上建立消息传递系统,包括配置STOMP代理中继、处理客户端消息、发送消息到客户端以及为目标用户发送消息。还讨论了处理消息异常的方法。





















1万+

被折叠的 条评论
为什么被折叠?



