關於spring 掃描不到jar中class文件的原因分析及解決
spring 掃描不到jar中class文件的原因及解決
背景
公司一web項目使用的是spring mvc開發的,老員工們寫瞭一個緩存service,即EhcacheService , 該緩存service在web中使用瞭spring 的@Scheduled 啟動加載緩存,代碼如下:
applicationContext.xml
<context:component-scan base-package="cn.com.service" />
EhcacheService .java
// 啟動加載緩存, 以上一次執行完為準 @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("++++++++++++++++++++緩存加載結束,耗時:" + (end - start) + "++++++++++++++++++++"); }
然而最近同步數據,需要用到EhcacheService , 本人也懶得重寫裡面的方法,便想著使用ClassPathXmlApplicationContext 或者FileSystemXmlApplicationContext 或者GenericXmlApplicationContext來加載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的數據量" + context.getBeanDefinitionNames().length); EhcacheService ehcacheService = (EhcacheService) context.getBean("ehcacheService"); }
我們先看一下在Eclipse中運行情況
我們再看看打包成Runable jar File後,使用 java -jar my.jar的運行情況
根據圖片中的錯誤,我們可以看到,spring-*.xml是成功被加載瞭,然而找不到bean, 很明顯,它存在一種可能,那就是bean的class文件沒有被spring掃描到。
那麼為什麼會出現這種情況呢?經過我多方面的查證,spring 掃描bean文件是通過Thread.currentThread().getContextClassLoader().getResource(packageName)加載的。那麼我們分析一下ContextClassLoader資源加載機制。
舉例說明:我們有這樣的一個: cn.com.Test, 類加載器首先會把這個包名轉化成文件夾的形式 cn/com, 然後到這個文件夾裡去加載Test.class。
然後,當你打包成Runable jar File時,jar的包和文件系統中的包便不是一個概念瞭,它不能將cn.com轉換成cn/com文件夾方式去解讀, 類加載轉換成cn/com去加載類的時候,便會報出classNotFoundException異常
下面我們使用如下代碼驗證一下這個過程:
// 項目中jar包所在物理路徑 String jarName = "C:\\Users\\Administrator\\Desktop\\my.jar"; // 項目中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()); }
打印結果如下:
META-INF/MANIFEST.MF cn/com/server/action/JobAction.class cn/com/server/annotation/DataDigestAnnotation.class cn/com/server/dao/EhcacheDao.class
然後我們打包成war包再看看他的war包物理路徑,我們可以看到打印結果如下:
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: 那麼我們怎麼解決spring 掃描不到jar中class這個問題呢?
A: 有一種做法,就是打jar包的時候,打成JAR file, 然後選擇 add directory entries, 如圖:
然後這種打包方式,雖然能解決spring 掃描不到jar中class文件問題,但是打並不是我們想要的,我們想要的是一個可以執行jar,也就是Runable JAR FILE。
Q: 那麼我們怎麼打包成Runable JAR FILE,並且解決spring 掃描不到jar中class的問題?
A:
1、首先使用Eclipse打包,打包成JAR file。
2、上傳到Linux, 解壓my.jar
unzip my.jar -d myapp
3、進入 myapp文件夾, 使用以下命令:
java -Djava.ext.dirs=WebContent/WEB-INF/lib cn.com.test
大功告成
其他技巧:除瞭上訴使用代碼方式查看jar包物理路徑,我們還可以是 jar tr my.jar來查看。如圖:
@ComponentScan註解進行掃描的幾種方式
方式一:掃描包
返回是String的數組,所以可是多個包路徑!也可是一個包路徑!完整寫法是
- 單個:@ComponentScan(basePackages = “xxx”)
- 多個:@ComponentScan(basePackages = {“xxx”,“aaa”,“…”})
註意:可以省略“basePackages =”
方式二:掃描類
同樣返回是String的數組,所以可以是有多個類名! 也可是一個類名!
- 單個:@ComponentScan(basePackageClasses = “”)
- 多個:@ComponentScan(basePackageClasses = {“xxx”,“aaa”,“…”})
註意:不可以省略“basePackageClasses =”
測試:
方式三:掃描包(通配式:開發常用)
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- 解決Spring boot 整合Junit遇到的坑
- springboot 啟動如何排除某些bean的註入
- spring IOC控制反轉原理詳解
- Spring框架學習之Spring @Autowired實現自動裝配的代碼
- 動態上傳jar包熱部署的實戰詳解