websocket使用@Autowired、@Value获取值为null解决方法

本文讲述了在使用Spring框架时,由于WebSocket的多实例特性导致@Autowired和@Value注解无法正常工作的问题。作者通过分析问题原因,提出将注入对象设为静态类级别变量,并在方法上使用set方法初始化,从而解决注入null的问题。此解决方案适用于处理Spring单例与WebSocket多实例冲突的场景。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

作为一名资深码农,原本以为闭着眼睛都可以用@Autowired、@Value轻松获取注入的对象,不料最近在写一个小工具时碰了一鼻子灰,先上代码:

@ServerEndpoint("/log")
@Component
public class LogWebSocketHandle {

    @Value("${wslog_commond}")//
    private  String wslog_commond1;

相信java 码农们再熟悉不过了,不过你不要高兴太早,取值时候一个null足以让你抓狂。

注意一个细节开头那段@ServerEndpoint表示这是一个websocket服务端类,根本原因在于:spring管理的都是单例(singleton)和 websocket (多对象)相冲突。websocket多用于客户端与服务端快速通信:聊天、实时输出日志等场景,这意味着每一个客户端请求服务端都会生成一个新的websocket实例,spring 仅仅针对@component、@controller注解完成单例模式管理任务(注意仅仅这一次、这一个对象赋值了)后续随着客户单端访问 websocket不断进行实例化(这些对象不是由spring管理的,所以启动过程并无法给后续的实际处理websocket会话的实例赋值)。不改变方法的话每个 websocket 对象的 注入对象值得都是 null。

了解了这点后,我们可以将要注入的对象设为类级别的变量,并通过在方法上使用@Autowired、@Value对变量进行初始化,进而使变量不为空:

//@Value("${wslog_commond}")千万不可打开这个注解
private static String wslog_commond;//static为关键

@Value("${wslog_commond}")
public void setWslog_commond(String wslog_commond) {

    LogWebSocketHandle.wslog_commond = wslog_commond;
}

总结:websocket与spring框架的@component、@controller无冲突,spring会按照原来规则实例化一个对象,只不过该对象在运行期间一直不会被使用,这就成了废物。将类的要注入属性改为静态类级别变量,通过第一次实例化调用set方法完成对类的变量赋值动作,变废为宝完成其使命。

本案例还是来自本人写的推docker送镜像到haobor工具,源码地址GitHub - kakahu2015/harbor-demo 

### 集成WebSocket并正确注入Bean #### 导入WebSocket依赖 为了在Spring Boot项目中集成WebSocket,首先需要引入必要的依赖项。这可以通过修改`pom.xml`文件来完成,在其中加入WebSocket的支持库[^2]。 ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-websocket</artifactId> </dependency> ``` #### 创建WebSocket配置类 接着定义一个用于配置WebSocket行为的Java类。此类通常会继承自`AbstractSessionWebSocketMessageBrokerConfigurer`或者实现`WebSocketMessageBrokerConfigurer`接口,并重写相应的方法以定制化WebSocket的功能设置。 #### 实现消息处理端点 创建带有`@ServerEndpoint`注解的服务端点类,该类负责监听特定路径上的连接请求以及接收来自客户端的消息。需要注意的是,在这个类里面可以直接通过构造函数参数列表的方式自动装配所需的Spring管理对象;然而有时可能会遇到无法成功注入的情况,这时可以考虑采用静态上下文工具类动态获取所需bean实例作为替代方案[^3]。 对于因某些原因导致的标准DI机制失效的情形下,可以在检测到成员变量为空时手动从应用上下文中出对应的bean: ```java if(this.webSocketStore == null){ this.webSocketStore = (WebSocketStore) SpringContextUtil.getBean("webSocketStore"); } ``` 这种方法虽然不是最优雅的选择,但在特殊场景下确实能够解决问题[^4]。 #### 完整示例代码片段 下面给出一段完整的代码样例,展示了如何在一个基于Spring Boot的应用程序里启用WebSocket支持并且安全可靠地执行Bean注入操作。 ```java // WebSocketConfig.java @Configuration @EnableWebSocketMessageBroker public class WebSocketConfig implements WebSocketMessageBrokerConfigurer { @Override public void configureMessageBroker(MessageBrokerRegistry config) { config.enableSimpleBroker("/topic"); config.setApplicationDestinationPrefixes("/app"); } @Override public void registerStompEndpoints(StompEndpointRegistry registry) { registry.addEndpoint("/ws").withSockJS(); } } // MyWebSocketHandler.java @Component @ServerEndpoint(value="/chat/{userId}") public class MyWebSocketHandler { private final Logger logger = LoggerFactory.getLogger(MyWebSocketHandler.class); // 使用@Autowired进行标准的依赖注入尝试 @Autowired private SomeService someService; // 如果上述方式不奏效,则可在此处提供备用加载逻辑 private transient volatile SomeService fallbackSomeService; @OnOpen public void onOpen(Session session, @PathParam("userId") String userId) throws IOException{ if(someService==null && fallbackSomeService!=null){ synchronized(this){ if(fallbackSomeService==null){ ApplicationContext context=ApplicationContextProvider.getApplicationContext(); fallbackSomeService=context.getBean(SomeService.class); } } } // ...其余业务逻辑... } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

kakahu2015

只发干货,如果你觉得不干请别打

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

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

打赏作者

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

抵扣说明:

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

余额充值