Spring5高级编程------Spring-WebSocket中STOMP配置的加载机制

本文深入探讨Spring WebSocket中STOMP配置的加载机制,详细解释@EnableWebSocketMessageBroker注解的作用,以及DelegatingWebSocketMessageBrokerConfiguration配置类如何实现对自定义配置的支持。

WebSocketMessageBrokerConfigurer配置类的加载

一、注解@EnableWebSocketMessageBroker

将这个注释添加到一个@Configuration类中,这样就可以使用更高级的消息传递子协议在WebSocket上启用代理支持的消息传递。@EnableWebSocketMessageBroker注解的主要作用就是引入一个配置类DelegatingWebSocketMessageBrokerConfiguration

package org.springframework.web.socket.config.annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import org.springframework.context.annotation.Import;

/**
 * Add this annotation to an {@code @Configuration} class to enable broker-backed
 * messaging over WebSocket using a higher-level messaging sub-protocol.
 *
 */
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
@Import(DelegatingWebSocketMessageBrokerConfiguration.class)
public @interface EnableWebSocketMessageBroker {

}

将这个注解放在自定义的WebSocketMessageBrokerConfigurer实现类上即可实现STOMP配置的自动加载,自定义STOMP配置类如下(@Configuration注解不是必须的,也可以使用@Import注解直接在主配置类中引入该配置类):

import org.springframework.web.socket.config.annotation.EnableWebSocketMessageBroker;
import org.springframework.context.annotation.Configuration;

@EnableWebSocketMessageBroker
@Configuration
public class WebSocketStompConfiguration implements WebSocketMessageBrokerConfigurer {
}

注意:WebSocketMessageBrokerConfigurer有一个空实现的抽象类AbstractWebSocketMessageBrokerConfigurer,由于WebSocketMessageBrokerConfigurer接口提供了默认实现,所以AbstractWebSocketMessageBrokerConfigurer已被废弃,这里不建议使用继承AbstractWebSocketMessageBrokerConfigurer的方式配置。

二、DelegatingWebSocketMessageBrokerConfiguration配置类

@EnableWebSocketMessageBroker 注解使用@Import()注解引入了该配置类源码如下:

package org.springframework.web.socket.config.annotation;

import java.util.ArrayList;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.messaging.converter.MessageConverter;
import org.springframework.messaging.handler.invocation.HandlerMethodArgumentResolver;
import org.springframework.messaging.handler.invocation.HandlerMethodReturnValueHandler;
import org.springframework.messaging.simp.config.ChannelRegistration;
import org.springframework.messaging.simp.config.MessageBrokerRegistry;
import org.springframework.util.CollectionUtils;
/**
 * A {@link WebSocketMessageBrokerConfigurationSupport} extension that detects
 * beans of type {@link WebSocketMessageBrokerConfigurer} and delegates to all
 * of them allowing callback style customization of the configuration provided
 * in {@link WebSocketMessageBrokerConfigurationSupport}.
 *
 * <p>This class is typically imported via {@link EnableWebSocketMessageBroker}.
 *
 * @author Rossen Stoyanchev
 * @since 4.0
 */
@Configuration
public class DelegatingWebSocketMessageBrokerConfiguration extends WebSocketMessageBrokerConfigurationSupport {
	private final List<WebSocketMessageBrokerConfigurer> configurers = new ArrayList<>();
	
	@Autowired(required = false)
	public void setConfigurers(List<WebSocketMessageBrokerConfigurer> configurers) {
		if (!CollectionUtils.isEmpty(configurers)) {
			this.configurers.addAll(configurers);
		}
	}
	
	@Override
	protected void registerStompEndpoints(StompEndpointRegistry registry) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.registerStompEndpoints(registry);
		}
	}
	
	@Override
	protected void configureWebSocketTransport(WebSocketTransportRegistration registration) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.configureWebSocketTransport(registration);
		}
	}

	@Override
	protected void configureClientInboundChannel(ChannelRegistration registration) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.configureClientInboundChannel(registration);
		}
	}

	@Override
	protected void configureClientOutboundChannel(ChannelRegistration registration) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.configureClientOutboundChannel(registration);
		}
	}

	@Override
	protected void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.addArgumentResolvers(argumentResolvers);
		}
	}

	@Override
	protected void addReturnValueHandlers(List<HandlerMethodReturnValueHandler> returnValueHandlers) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.addReturnValueHandlers(returnValueHandlers);
		}
	}

	@Override
	protected boolean configureMessageConverters(List<MessageConverter> messageConverters) {
		boolean registerDefaults = true;
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			if (!configurer.configureMessageConverters(messageConverters)) {
				registerDefaults = false;
			}
		}
		return registerDefaults;
	}

	@Override
	protected void configureMessageBroker(MessageBrokerRegistry registry) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.configureMessageBroker(registry);
		}
	}

}

这个配置类的第一个作用是发现并收集Spring容器中的所有WebSocketMessageBrokerConfigurer类型的配置类,代码如下:

private final List<WebSocketMessageBrokerConfigurer> configurers = new ArrayList<>();
	
	@Autowired(required = false)
	public void setConfigurers(List<WebSocketMessageBrokerConfigurer> configurers) {
		if (!CollectionUtils.isEmpty(configurers)) {
			this.configurers.addAll(configurers);
		}
	}

这个配置类的第二个作用是实现父类的8个方法,在这8个方法中会遍历获取所有WebSocketMessageBrokerConfigurer 类并分别调用对应的配置方法,从而引入用户自定义的配置,代码如下(这里只列举一个,其他7个类似):

@Override
	protected void registerStompEndpoints(StompEndpointRegistry registry) {
		for (WebSocketMessageBrokerConfigurer configurer : this.configurers) {
			configurer.registerStompEndpoints(registry);
		}
	}

三、WebSocketMessageBrokerConfigurationSupport抽象类

WebSocketMessageBrokerConfigurationSupport是DelegatingWebSocketMessageBrokerConfiguration的抽象父类。其中包含2个需要DelegatingWebSocketMessageBrokerConfiguration实现的方法:

public abstract class WebSocketMessageBrokerConfigurationSupport extends AbstractMessageBrokerConfiguration {

	protected void configureWebSocketTransport(WebSocketTransportRegistration registry) {
	}
	
	protected abstract void registerStompEndpoints(StompEndpointRegistry registry);
	}

四、AbstractMessageBrokerConfiguration抽象类

AbstractMessageBrokerConfiguration是DelegatingWebSocketMessageBrokerConfiguration的抽象父类。其中包含6个需要DelegatingWebSocketMessageBrokerConfiguration实现的方法:

public abstract class AbstractMessageBrokerConfiguration implements ApplicationContextAware {

	protected void configureClientInboundChannel(ChannelRegistration registration) {
		}
		
	protected void configureClientOutboundChannel(ChannelRegistration registration) {
	}
	
	protected void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {
	}
	
	protected void addReturnValueHandlers(List<HandlerMethodReturnValueHandler> returnValueHandlers) {
	}
	
	protected boolean configureMessageConverters(List<MessageConverter> messageConverters) {
		return true;
	}
	
	protected void configureMessageBroker(MessageBrokerRegistry registry) {
	}
}

五、STOMP配置中的三个关键接口

1、StompEndpointRegistry

2、StompWebSocketEndpointRegistration

3、SockJsServiceRegistration

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

豢龙先生

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

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

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

打赏作者

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

抵扣说明:

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

余额充值