WebSocket无法@Autowired自动注入
前言
最近在项目中需要用到WebSocket,当我跟SpringBoot整合,然后做一些持久层的相关操作的时候,突然空指针异常,原因在于@Autowired或者@Resource无法自动注入,一开始以为是我太久没写后端代码,整合方式出了问题(QAQ),改了一堆配置还是没解决,最后写了个普通的Controller出来测试,发现完全OJBK。所以问题的源头终于找到了,那就是WebSocket在搞鬼!
原因
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。
解决
传入applicationContext 参数
@EnableDiscoveryClient//注册服务
@SpringBootApplication(exclude = DataSourceAutoConfiguration.class)
@ComponentScan(basePackages = "com.example") //swagger扫描文件
@EnableFeignClients(basePackages = "com.example")
public class ServiceChatApplication {
public static void main(String[] args) {
ConfigurableApplicationContext applicationContext = SpringApplication.run(ServiceChatApplication.class, args);
WebSocketServerController.setApplicationContext(applicationContext);
}
}
private static ApplicationContext applicationContext;
public static void setApplicationContext(ApplicationContext applicationContext) {
WebSocketServerController.applicationContext = applicationContext;
}
WebSocketService webSocketService = applicationContext.getBean(WebSocketService.class);
UserInfo userInfo = webSocketService.getUserInfo(userId);
gateway无法加载配置文件
今天要开发客服模块,加载完前端之后发现有问题一开始以为是前端导入新的依赖有问题,回滚之后问题还是有,这里排查到应该是无法网关无法使用,在尝试重启idea之后还是没有用
这时打开idea生成的文件发现里面没有配置信息,此时gateway无法加载信息
这时发现resources文件标记不对
导入第三方文件出现错误
导入的时候注意版本号
npm install less-loader@4.1.0 -D
npm install less@3.9.0 -D
后端模块无法启动
Failed to register @ServerEndpoint class: class com.example.yygh.chat.contro
websocket不能走网关
写完后端模块之后发现他没有请求到聊天模块,在gateway这里报错,这里猜想一个是不能通过websocket
WebSocket connection to ‘ws://192.168.17.212:8080/ws‘ failed:
这个问题困扰了我一天发现一定要加一个service
解决报错:WebSocket connection to ‘ws://192.168.17.212:8080/ws‘ failed: