问题:
使用Springboot中websocket的功能的时候,导致其中一个service自动注入结果为null,定位到问题之后,,发现有一个类没有完成注入。
解决
spring 或 springboot 的 websocket 里面使用 @Autowired 注入 service 或 bean 时,报空指针异常,service 为 null(并不是不能被注入)。
解决方法:将要注入的 service 改成 static,就不会为null了。
@Controller
@ServerEndpoint(value="/chatSocket")
public class ChatSocket {
// 这里使用静态,让 service 属于类
private static ChatService chatService;
// 注入的时候,给类的 service 注入
@Autowired
public void setChatService(ChatService chatService) {
ChatSocket.chatService = chatService;
}
}
本质原因:
spring管理的都是单例(singleton),和 websocket (多对象)相冲突。
详细解释:项目启动时初始化,会初始化 websocket (非用户连接的),spring 同时会为其注入 service,该对象的 service 不是 null,被成功注入。但是,由于 spring 默认管理的是单例,所以只会注入一次 service。当新用户进入聊天时,系统又会创建一个新的 websocket 对象,这时矛盾出现了:spring 管理的都是单例,不会给第二个 websocket 对象注入 service,所以导致只要是用户连接创建的 websocket 对象,都不能再注入了。
像 controller 里面有 service, service 里面有 dao。因为 controller,service ,dao 都有是单例,所以注入时不会报 null。但是 websocket 不是单例,所以使用spring注入一次后,后面的对象就不会再注入了,会报null。
原文出处:https://www.cnblogs.com/dyingstraw/p/12865800.html#:~:text=spring%20%E6%88%96%20springboot%20%E7%9A%84%20websocket%20%E9%87%8C%E9%9D%A2%E4%BD%BF%E7%94%A8%20%40Autowired%20%E6%B3%A8%E5%85%A5,bean%20%E6%97%B6%EF%BC%8C%E6%8A%A5%E7%A9%BA%E6%8C%87%E9%92%88%E5%BC%82%E5%B8%B8%EF%BC%8Cservice%20%E4%B8%BA%20null%EF%BC%88%E5%B9%B6%E4%B8%8D%E6%98%AF%E4%B8%8D%E8%83%BD%E8%A2%AB%E6%B3%A8%E5%85%A5%EF%BC%89%E3%80%82%20%E8%A7%A3%E5%86%B3%E6%96%B9%E6%B3%95%EF%BC%9A%E5%B0%86%E8%A6%81%E6%B3%A8%E5%85%A5%E7%9A%84%20service%20%E6%94%B9%E6%88%90%20static%EF%BC%8C%E5%B0%B1%E4%B8%8D%E4%BC%9A%E4%B8%BAnull%E4%BA%86%E3%80%82