解決SpringBoot webSocket 資源無法加載、tomcat啟動(dòng)報(bào)錯(cuò)的問題
問題描述:
1. 項(xiàng)目集成WebSocket,且打包發(fā)布tomcat時(shí)出現(xiàn)websocket is already in CLOSING or CLOSE state這樣的問題,建議參考“解決方法二”,但是“解決方法一”請(qǐng)要了解查看 ,因?yàn)榻鉀Q方法二是在一的基礎(chǔ)上進(jìn)行更正
2. 如果出現(xiàn)javax.websocket.server.ServerContainer not available這樣的錯(cuò)誤,請(qǐng)參考“解決方法一”中步驟3
解決方法一:(常規(guī)解決、不推薦)
步驟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接受消息的服務(wù)類
步驟2:修改WebSocketServer
private static ManageUserMapper manageUserMapper;
public static void setApplicationContext(ConfigurableApplicationContext applicationContext) {
WebSocketServer.manageUserMapper = applicationContext.getBean(ManageUserMapper.class);
}
步驟3: 修改pom.xml
由于我們?cè)陂_發(fā)過程中,如果按照以上步驟1、2進(jìn)行修改,一般不會(huì)出現(xiàn)問題,
但是如果我們打包發(fā)布tomcat,就會(huì)出現(xiàn):javax.websocket.server.ServerContainer not available這樣的錯(cuò)誤,步驟3為常規(guī)解決一下問題方式
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> <!-- 去除內(nèi)嵌tomcat --> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-tomcat</artifactId> </exclusion> </exclusions> </dependency>
需要去除web-starter的tomcat,因?yàn)槿绻蝗コ龝?huì)導(dǎo)致沖突,
如果出現(xiàn)這種問題,你還需要在websocketConfig中做如下修改:(websocket為自定義配置類)
/**
* 服務(wù)器節(jié)點(diǎn)
*
* 如果使用獨(dú)立的servlet容器,而不是直接使用springboot的內(nèi)置容器,就不要注入ServerEndpointExporter,因?yàn)樗鼘⒂扇萜髯约禾峁┖凸芾?
* @return
*/
@Bean
@Profile("dev")
public ServerEndpointExporter serverEndpointExporter() {
return new ServerEndpointExporter();
}
在application.properties做如下配置
##prod:生產(chǎn)環(huán)境 dev:開發(fā)環(huán)境(WINDOWS) spring.profiles.active=dev
這里加入了@Profile("dev") 這個(gè)注解。意思是在開發(fā)的過程中去調(diào)用
.記住如果開發(fā)過程中,記得一定要把pom.xml中的去除tomcat那句話給注釋掉,上線才需要去除
問題反思:(為什么不建議這么解決問題)
這種方式確實(shí)可以常規(guī)解決websocket打包tomcat不報(bào)錯(cuò)問題,同時(shí)也解決了在資源無法加載的問題,但是這樣卻十分的麻煩,理由如下:
1. 繁瑣:生產(chǎn)環(huán)境和開發(fā)環(huán)境要一直切換是否注釋tomcat
2. 局限性大:我們?cè)贐ootApplication中添加了websocketserver去訪問資源的語句,但是其他地方難道沒有需要調(diào)用的嗎,如果有,難道像這樣一直累計(jì)添加下去,每個(gè)server設(shè)置一下
核心思想是資源無法加載的問題,如果你有寫線程,且線程你也調(diào)用了數(shù)據(jù)庫資源,那么解決方法是一致,編寫一個(gè)公用的工具類解決
解決方法二:(推薦)
步驟1:新建一個(gè)工具類,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去調(diào)用資源了,一步到位,大家都能用,需要的時(shí)候直接通過getbeanTool去請(qǐng)求就行了
步驟3:解決tomcat開發(fā)環(huán)境,生產(chǎn)環(huán)境pom.xml中注釋和打開內(nèi)置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 ,同時(shí)重寫了configure方法。采用這種寫法,如論是開發(fā)還是測(cè)試,你都不用管是否需要注釋內(nèi)置的tomcat了,是不是很方便,至于原理,這里不解釋,篇幅太長(zhǎng),自己去百度吧
總結(jié):
1. 最近做了一個(gè)簡(jiǎn)單的IM功能,遇到了這些問題,作為記錄,防止哪天我忘了
2. 網(wǎng)上解決方法的可能有用,但是寫的不明不白,不友好,而且不見得好用,在解決問題的時(shí)候一定要考慮到通用性,開發(fā)一定要為了便利,不然坑了你也坑了被人,改這改那誰尼瑪記得住
3. 此博文為本人自身角度進(jìn)行編寫,作為記錄。如對(duì)他人有幫助,可以點(diǎn)個(gè)贊,本人不回復(fù),不評(píng)論。
到此這篇關(guān)于解決SpringBoot webSocket 資源無法加載、tomcat啟動(dòng)報(bào)錯(cuò)的問題的文章就介紹到這了,更多相關(guān)SpringBoot webSocket 資源無法加載內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
在SpringBoot中使用JWT的實(shí)現(xiàn)方法
這篇文章主要介紹了在SpringBoot中使用JWT的實(shí)現(xiàn)方法,詳細(xì)的介紹了什么是JWT和JWT實(shí)戰(zhàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-12-12
基于java線程池讀取單個(gè)SQL數(shù)據(jù)庫表
這篇文章主要為大家詳細(xì)介紹了基于java線程池讀取單個(gè)SQL數(shù)據(jù)庫表,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-08-08
java多線程編程之使用runnable接口創(chuàng)建線程
實(shí)現(xiàn)Runnable接口的類必須使用Thread類的實(shí)例才能創(chuàng)建線程,通過Runnable接口創(chuàng)建線程分為以下兩步2014-01-01
一文教會(huì)Java新手使用Spring?MVC中的查詢字符串和查詢參數(shù)
在使用springMVC框架構(gòu)建web應(yīng)用,客戶端常會(huì)請(qǐng)求字符串、整型、json等格式的數(shù)據(jù),這篇文章主要給大家介紹了關(guān)于通過一文教會(huì)Java新手使用Spring?MVC中的查詢字符串和查詢參數(shù)的相關(guān)資料,需要的朋友可以參考下2024-01-01
ElasticSearch創(chuàng)建后索引修改數(shù)據(jù)類型方法步驟
Elasticsearch存儲(chǔ)數(shù)據(jù)之前需要先創(chuàng)建索引,類似于結(jié)構(gòu)型數(shù)據(jù)庫建庫建表,創(chuàng)建索引時(shí)定義了每個(gè)字段的索引方式和數(shù)據(jù)類型,這篇文章主要給大家介紹了關(guān)于ElasticSearch創(chuàng)建后索引修改數(shù)據(jù)類型的方法步驟,需要的朋友可以參考下2023-09-09
Java中關(guān)于控制臺(tái)讀取數(shù)字或字符串的方法
下面小編就為大家?guī)硪黄狫ava中關(guān)于控制臺(tái)讀取數(shù)字或字符串的方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10
Springboot項(xiàng)目實(shí)現(xiàn)Mysql多數(shù)據(jù)源切換的完整實(shí)例
這篇文章主要給大家介紹了關(guān)于Springboot項(xiàng)目實(shí)現(xiàn)Mysql多數(shù)據(jù)源切換的完整實(shí)例,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
Java實(shí)現(xiàn)儲(chǔ)存對(duì)象并按對(duì)象某屬性排序的幾種方法示例
這篇文章主要介紹了Java實(shí)現(xiàn)儲(chǔ)存對(duì)象并按對(duì)象某屬性排序的幾種方法,結(jié)合實(shí)例形式詳細(xì)分析了Java儲(chǔ)存對(duì)象并按對(duì)象某屬性排序的具體實(shí)現(xiàn)方法與操作注意事項(xiàng),需要的朋友可以參考下2020-05-05

