關(guān)于spring 掃描不到j(luò)ar中class文件的原因分析及解決
spring 掃描不到j(luò)ar中class文件的原因及解決
背景
公司一web項(xiàng)目使用的是spring mvc開發(fā)的,老員工們寫了一個(gè)緩存service,即EhcacheService , 該緩存service在web中使用了spring 的@Scheduled 啟動(dòng)加載緩存,代碼如下:
applicationContext.xml
<context:component-scan base-package="cn.com.service" />
EhcacheService .java
// 啟動(dòng)加載緩存, 以上一次執(zhí)行完為準(zhǔn) @Scheduled(fixedDelay = 365 * 24 * 60 * 60 * 1000) public void initEhcache() { logger.debug("++++++++++++++++++++緩存加載開始++++++++++++++++++++"); long start = System.currentTimeMillis(); try { this.ehcacheService.loadCache(); } catch (Exception e) { logger.error(e.getMessage(), e); } long end = System.currentTimeMillis(); logger.debug("++++++++++++++++++++緩存加載結(jié)束,耗時(shí):" + (end - start) + "++++++++++++++++++++"); }
然而最近同步數(shù)據(jù),需要用到EhcacheService , 本人也懶得重寫里面的方法,便想著使用ClassPathXmlApplicationContext 或者FileSystemXmlApplicationContext 或者GenericXmlApplicationContext來(lái)加載spring配置,然后打包成jar包,丟到linux上,使用java -jar my.jar。代碼如下:
test.java
// 程序入口 public static void main(String[] args) throws Exception { // 加載spring配置 GenericXmlApplicationContext context = new GenericXmlApplicationContext(); context.setValidating(false); context.load("classpath*:spring-*.xml"); context.refresh(); // ApplicationContext ctx = new FileSystemXmlApplicationContext("spring-*.xml"); // ApplicationContext context = new ClassPathXmlApplicationContext("spring-*.xml"); System.out.println("bean的數(shù)據(jù)量" + context.getBeanDefinitionNames().length); EhcacheService ehcacheService = (EhcacheService) context.getBean("ehcacheService"); }
我們先看一下在Eclipse中運(yùn)行情況
我們?cè)倏纯创虬蒖unable jar File后,使用 java -jar my.jar的運(yùn)行情況
根據(jù)圖片中的錯(cuò)誤,我們可以看到,spring-*.xml是成功被加載了,然而找不到bean, 很明顯,它存在一種可能,那就是bean的class文件沒有被spring掃描到。
那么為什么會(huì)出現(xiàn)這種情況呢?經(jīng)過我多方面的查證,spring 掃描bean文件是通過Thread.currentThread().getContextClassLoader().getResource(packageName)加載的。那么我們分析一下ContextClassLoader資源加載機(jī)制。
舉例說(shuō)明:我們有這樣的一個(gè): cn.com.Test, 類加載器首先會(huì)把這個(gè)包名轉(zhuǎn)化成文件夾的形式 cn/com, 然后到這個(gè)文件夾里去加載Test.class。
然后,當(dāng)你打包成Runable jar File時(shí),jar的包和文件系統(tǒng)中的包便不是一個(gè)概念了,它不能將cn.com轉(zhuǎn)換成cn/com文件夾方式去解讀, 類加載轉(zhuǎn)換成cn/com去加載類的時(shí)候,便會(huì)報(bào)出classNotFoundException異常
下面我們使用如下代碼驗(yàn)證一下這個(gè)過程:
// 項(xiàng)目中jar包所在物理路徑 String jarName = "C:\\Users\\Administrator\\Desktop\\my.jar"; // 項(xiàng)目中war包所在物理路徑 //String jarName = "C:\\Users\\Administrator\\Desktop\\my.war"; JarFile jarFile = new JarFile(jarName); Enumeration<JarEntry> entrys = jarFile.entries(); while (entrys.hasMoreElements()) { JarEntry jarEntry = entrys.nextElement(); System.out.println(jarEntry.getName()); }
打印結(jié)果如下:
META-INF/MANIFEST.MF cn/com/server/action/JobAction.class cn/com/server/annotation/DataDigestAnnotation.class cn/com/server/dao/EhcacheDao.class
然后我們打包成war包再看看他的war包物理路徑,我們可以看到打印結(jié)果如下:
META-INF/MANIFEST.MF META-INF/ WEB-INF/classes/ WEB-INF/classes/cn/ WEB-INF/classes/cn/com/ WEB-INF/classes/cn/com/ WEB-INF/classes/cn/com/server/ WEB-INF/classes/cn/com/server/action/ WEB-INF/classes/cn/com/server/action/JobAction.class WEB-INF/classes/cn/com/server/addrsrv/ WEB-INF/classes/cn/com/server/addrsrv/GeoAddrSrv.class …..
我們可以看到war類的文件目錄和jar的文件目錄明顯不同,這樣就能解釋上面我所描述的問題。
Q: 那么我們?cè)趺唇鉀Qspring 掃描不到j(luò)ar中class這個(gè)問題呢?
A: 有一種做法,就是打jar包的時(shí)候,打成JAR file, 然后選擇 add directory entries, 如圖:
然后這種打包方式,雖然能解決spring 掃描不到j(luò)ar中class文件問題,但是打并不是我們想要的,我們想要的是一個(gè)可以執(zhí)行jar,也就是Runable JAR FILE。
Q: 那么我們?cè)趺创虬蒖unable JAR FILE,并且解決spring 掃描不到j(luò)ar中class的問題?
A:
1、首先使用Eclipse打包,打包成JAR file。
2、上傳到Linux, 解壓my.jar
unzip my.jar -d myapp
3、進(jìn)入 myapp文件夾, 使用以下命令:
java -Djava.ext.dirs=WebContent/WEB-INF/lib cn.com.test
大功告成
其他技巧:除了上訴使用代碼方式查看jar包物理路徑,我們還可以是 jar tr my.jar來(lái)查看。如圖:
@ComponentScan注解進(jìn)行掃描的幾種方式
方式一:掃描包
返回是String的數(shù)組,所以可是多個(gè)包路徑!也可是一個(gè)包路徑!完整寫法是
- 單個(gè):@ComponentScan(basePackages = “xxx”)
- 多個(gè):@ComponentScan(basePackages = {“xxx”,“aaa”,“…”})
注意:可以省略“basePackages =”
方式二:掃描類
同樣返回是String的數(shù)組,所以可以是有多個(gè)類名! 也可是一個(gè)類名!
- 單個(gè):@ComponentScan(basePackageClasses = “”)
- 多個(gè):@ComponentScan(basePackageClasses = {“xxx”,“aaa”,“…”})
注意:不可以省略“basePackageClasses =”
測(cè)試:
方式三:掃描包(通配式:開發(fā)常用)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Java中將 int[] 數(shù)組 轉(zhuǎn)換為 List分享
這篇文章主要介紹了Java中將 int[] 數(shù)組 轉(zhuǎn)換為 List分享的相關(guān)資料,需要的朋友可以參考下2022-12-12rabbitmq學(xué)習(xí)系列教程之消息應(yīng)答(autoAck)、隊(duì)列持久化(durable)及消息持久化
這篇文章主要介紹了rabbitmq學(xué)習(xí)系列教程之消息應(yīng)答(autoAck)、隊(duì)列持久化(durable)及消息持久化,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-03-03SpringBoot讀取資源目錄中JSON文件的方法實(shí)例
最近做項(xiàng)目遇到需要將json類型的配置文件引用到項(xiàng)目中,已經(jīng)將讀取json文件的方法封裝成工具類,下面這篇文章主要給大家介紹了關(guān)于SpringBoot讀取資源目錄中JSON文件的相關(guān)資料,需要的朋友可以參考下2023-04-04java this 用法詳解及簡(jiǎn)單實(shí)例
這篇文章主要介紹了java this 用法詳解及簡(jiǎn)單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-03-03Kafka?日志存儲(chǔ)實(shí)現(xiàn)過程
這篇文章主要為大家介紹了Kafka?日志存儲(chǔ)的實(shí)現(xiàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05