解决SpringBoot webSocket 资源无法加载、tomcat启动报错的问题
作者:fyhju1
问题描述:
1. 项目集成WebSocket,且打包发布tomcat时出现websocket is already in CLOSING or CLOSE state这样的问题,建议参考“解决方法二”,但是“解决方法一”请要了解查看 ,因为解决方法二是在一的基础上进行更正
2. 如果出现javax.websocket.server.ServerContainer not available这样的错误,请参考“解决方法一”中步骤3
解决方法一:(常规解决、不推荐)
步骤1:在BootApplication中修改:
public static void main(String[] args) { SpringApplication springApplication = new SpringApplication(BootApplication.class); springApplication.addListeners(new ApplicationPidFileWriter()); ConfigurableApplicationContext applicationContext = springApplication.run(args); WebSocketServer.setApplicationContext(applicationContext);
注:这里的WebSocketServer是指你自定义的websocket接受消息的服务类
步骤2:修改WebSocketServer
private static ManageUserMapper manageUserMapper; public static void setApplicationContext(ConfigurableApplicationContext applicationContext) { WebSocketServer.manageUserMapper = applicationContext.getBean(ManageUserMapper.class); }
步骤3: 修改pom.xml
由于我们在开发过程中,如果按照以上步骤1、2进行修改,一般不会出现问题,
但是如果我们打包发布tomcat,就会出现:javax.websocket.server.ServerContainer not available这样的错误,步骤3为常规解决一下问题方式
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <!-- 去除内嵌tomcat --> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency>
需要去除web-starter的tomcat,因为如果不去除会导致冲突,
如果出现这种问题,你还需要在websocketConfig中做如下修改:(websocket为自定义配置类)
/** * 服务器节点 * * 如果使用独立的servlet容器,而不是直接使用springboot的内置容器,就不要注入ServerEndpointExporter,因为它将由容器自己提供和管理 * @return */ @Bean @Profile("dev") public ServerEndpointExporter serverEndpointExporter() { return new ServerEndpointExporter(); }
在application.properties做如下配置
##prod:生产环境 dev:开发环境(WINDOWS) spring.profiles.active=dev
这里加入了@Profile("dev") 这个注解。意思是在开发的过程中去调用
.记住如果开发过程中,记得一定要把pom.xml中的去除tomcat那句话给注释掉,上线才需要去除
问题反思:(为什么不建议这么解决问题)
这种方式确实可以常规解决websocket打包tomcat不报错问题,同时也解决了在资源无法加载的问题,但是这样却十分的麻烦,理由如下:
1. 繁琐:生产环境和开发环境要一直切换是否注释tomcat
2. 局限性大:我们在BootApplication中添加了websocketserver去访问资源的语句,但是其他地方难道没有需要调用的吗,如果有,难道像这样一直累计添加下去,每个server设置一下
核心思想是资源无法加载的问题,如果你有写线程,且线程你也调用了数据库资源,那么解决方法是一致,编写一个公用的工具类解决
解决方法二:(推荐)
步骤1:新建一个工具类,getbeanTool
此工具类为之后加载资源的工具类,公用
@Component public class GetBeanTool implements ApplicationContextAware{ private static ApplicationContext applicationContext = null; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { GetBeanTool.applicationContext = applicationContext; } public static Object getBeanByName(String beanName) { if (applicationContext == null){ return null; } return applicationContext.getBean(beanName); } public static <T> T getBean(Class<T> type) { return applicationContext.getBean(type); } }
步骤2:在线程、websocket等server中,加载资源
以WebSocketServer作为参考,如下修改:
private ManageUserMapper manageUserMapper; public WebSocketServer() { this.manageUserMapper=GetBeanTool.getBean(ManageUserMapper.class); }
这么做你就不用在BootApplication中添加语句,来让你的server去调用资源了,一步到位,大家都能用,需要的时候直接通过getbeanTool去请求就行了
步骤3:解决tomcat开发环境,生产环境pom.xml中注释和打开内置tomcat的问题
这里需要修改BootApplication,如下修改:
@SpringBootApplication @Configuration public class BootApplication extends SpringBootServletInitializer { @Override protected SpringApplicationBuilder configure( SpringApplicationBuilder application) { return application.sources(BootApplication.class); } public static void main(String[] args) { SpringApplication.run(BootApplication.class, args); } }
可以看到这里我继承了SpringBootServletInitializer ,同时重写了configure方法。采用这种写法,如论是开发还是测试,你都不用管是否需要注释内置的tomcat了,是不是很方便,至于原理,这里不解释,篇幅太长,自己去百度吧
总结:
1. 最近做了一个简单的IM功能,遇到了这些问题,作为记录,防止哪天我忘了
2. 网上解决方法的可能有用,但是写的不明不白,不友好,而且不见得好用,在解决问题的时候一定要考虑到通用性,开发一定要为了便利,不然坑了你也坑了被人,改这改那谁尼玛记得住
3. 此博文为本人自身角度进行编写,作为记录。如对他人有帮助,可以点个赞,本人不回复,不评论。
到此这篇关于解决SpringBoot webSocket 资源无法加载、tomcat启动报错的问题的文章就介绍到这了,更多相关SpringBoot webSocket 资源无法加载内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!