解决SpringBoot webSocket 资源无法加载、tomcat启动报错的问题

网友投稿 268 2023-02-18

解决SpringBoot webSocket 资源无法加载、tomcat启动报错的问题

问题描述:

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 applXYHjVeuicationContext = 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为常规解决一下问题方式

org.springframework.boot

spring-boot-starter-web

org.springframework.boot

spring-boot-starter-tomcat

需要去除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 getBean(Class 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. 此博文为本人自身角度进行编写,作为记录。如对他人有帮助,可以点个赞,本人不回复,不评论。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:SpringBoot 利用thymeleaf自定义错误页面
下一篇:高并发api接口平台搭建(api高并发解决方案)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~