WebSocket类明明注入了Bean,为什么报错为null

在 WebSocket 类中注入 Bean 看似可行而注入 Bean 报错为null,通常是由于Spring 的单例管理机制与 WebSocket 多实例创建特性冲突导致的,具体分析如下:

原因分析

  • Spring 的单例特性 :Spring 默认以单例模式管理 Bean,即一个 Bean 在容器中只创建一次。项目启动时,会初始化一个 WebSocket 实例(非用户连接时),此时 Spring 会为该实例注入 Bean,该实例的 Bean 不会为null
  • WebSocket 的多实例创建 :当新用户连接时,系统会创建新的 WebSocket 实例。由于 Spring 的单例机制,不会为后续创建的 WebSocket 实例再注入 Bean,导致这些新实例中的 Bean 为null

解决方案

通过ApplicationContext手动获取 Bean,绕过 Spring 自动注入的单例限制,确保每个 WebSocket 实例能获取到 Bean。步骤如下:

  1. 创建 Spring 上下文工具类

    复制代码
    import org.springframework.beans.BeansException;
    import org.springframework.context.ApplicationContext;
    import org.springframework.context.ApplicationContextAware;
    import org.springframework.stereotype.Component;
    
    @Component
    public class SpringContextUtil implements ApplicationContextAware {
        private static ApplicationContext applicationContext;
    
        @Override
        public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
            SpringContextUtil.applicationContext = applicationContext;
        }
    
        public static ApplicationContext getApplicationContext() {
            return applicationContext;
        }
    
        public static <T> T getBean(Class<T> clazz) {
            return applicationContext.getBean(clazz);
        }
    }
  2. 在 WebSocket 类中手动获取 Bean

    复制代码
    import javax.websocket.Session;
    import javax.websocket.server.ServerEndpoint;
    import org.springframework.stereotype.Component;
    
    @Component
    @ServerEndpoint("/hot-search-ws")
    public class WebSocketServerSearch {
        private HotSearchService hotSearchService;
    
        @javax.websocket.OnOpen
        public void onOpen(Session session) {
            // 手动获取service
            hotSearchService = SpringContextUtil.getBean(HotSearchService.class);
            sendHotSearches(session);
        }
    
        private void sendHotSearches(Session session) {
            try {
                if (hotSearchService != null) {
                    List<HotSearch> randomHotSearches = hotSearchService.getRandomHotSearches(5);
                    String hotSearchList = randomHotSearches.stream()
                            .map(HotSearch::getText)
                            .collect(Collectors.joining("\n"));
                    session.getBasicRemote().sendText(hotSearchList);
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    通过上述方法,每个 WebSocket 实例在需要时主动从 Spring 上下文中获取 Bean,避免因单例注入机制导致的null问题。

相关推荐
一嘴一个橘子20 分钟前
spring-aop 的 基础使用 - 4 - 环绕通知 @Around
java
小毅&Nora37 分钟前
【Java线程安全实战】⑨ CompletableFuture的高级用法:从基础到高阶,结合虚拟线程
java·线程安全·虚拟线程
冰冰菜的扣jio37 分钟前
Redis缓存中三大问题——穿透、击穿、雪崩
java·redis·缓存
小璐猪头1 小时前
专为 Spring Boot 设计的 Elasticsearch 日志收集 Starter
java
ps酷教程1 小时前
HttpPostRequestDecoder源码浅析
java·http·netty
闲人编程1 小时前
消息通知系统实现:构建高可用、可扩展的企业级通知服务
java·服务器·网络·python·消息队列·异步处理·分发器
栈与堆1 小时前
LeetCode-1-两数之和
java·数据结构·后端·python·算法·leetcode·rust
OC溥哥9992 小时前
Paper MinecraftV3.0重大更新(下界更新)我的世界C++2D版本隆重推出,拷贝即玩!
java·c++·算法
星火开发设计2 小时前
C++ map 全面解析与实战指南
java·数据结构·c++·学习·算法·map·知识
*才华有限公司*2 小时前
RTSP视频流播放系统
java·git·websocket·网络协议·信息与通信