软件编程
位置:首页>> 软件编程>> java编程>> 解决SpringBoot webSocket 资源无法加载、tomcat启动报错的问题

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

作者:fyhju1  发布时间:2021-07-28 05:06:42 

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

来源:https://blog.csdn.net/fyhju1/article/details/110111135

0
投稿

猜你喜欢

手机版 软件编程 asp之家 www.aspxhome.com