溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何通過IDEA快速定位和排除依賴沖突問題

發(fā)布時間:2021-07-22 16:04:48 來源:億速云 閱讀:761 作者:小新 欄目:編程語言

小編給大家分享一下如何通過IDEA快速定位和排除依賴沖突問題,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

前言

我們程序員在開發(fā)的時候經(jīng)常會遇到各種各樣的 BUG 問題,其中大部分是業(yè)務(wù)邏輯異常,還有一些是代碼書寫不規(guī)范造成的異常例如:NullPointException(NPE),IndexOutOfBoundsException 等等,其實這些我們都好定位和修復(fù)。但是還有一些運行時異常定位起來是特別頭疼的,那就是 jar 包沖突引起的異常。

一般程序在運行時發(fā)生類似于 java.lang.ClassNotFoundException , Method not found: '......' ,或者莫名其妙的異常信息,這種情況一般很大可能就是 jar包依賴沖突的問題引起的了。

至于為什么會發(fā)生 jar包依賴沖突?這種問題大致可以歸納為如下幾個原因:

  • 版本不匹配,高版本依賴了低版本,或者低版本依賴了高版本。 例如引入第三方庫,但是第三方庫基于的是 JDK7,而你們項目使用的是JDK8。

  • 重復(fù)引入不同版本jar包,造成使用錯誤。 很多時候我們引入第三方輪子,它們依賴引入某個基礎(chǔ)工具使用的是 v 1.0 的 jar,但是我們項目中自己也引入了該 jar,但是版本是 v 2.3,這時就會造成項目中使用同一個組件但是依賴了兩個不同版本的jar,沖突就會發(fā)生。

可以看到,其實總的來說 jar 包沖突的主要原因就是依賴的版本沖突。

異常發(fā)生

項目中需要導(dǎo)出報表,技術(shù)選型的時候,一般是選用 Apache POI,但是 POI 的使用方式比較基礎(chǔ),開發(fā)量大,容易出現(xiàn)內(nèi)存溢出的問題。

考慮到阿里開源了一套解析和生成Excel的工具 - EasyExcel,具有避免內(nèi)存溢出OOM的情況發(fā)生,而且使用方便簡單,所以就將它引入到了我們的項目中,具體的使用版本是 1.0.2。

<dependency>
 <groupId>com.alibaba</groupId>
 <artifactId>easyexcel</artifactId>
 <version>1.0.2</version>
</dependency>

而另一個模塊需要使用 POI 的將 Word 轉(zhuǎn)成 PDF 的功能,所以同時又引入了如下 POI 的依賴:

<!-- poi utils -->
<dependency>
 <groupId>org.apache.poi</groupId>
 <artifactId>poi</artifactId>
 <version>3.15</version>
</dependency>
<dependency>
 <groupId>org.apache.poi</groupId>
 <artifactId>poi-ooxml</artifactId>
 <version>3.15</version>
</dependency>

我們從 Maven Repository 可以發(fā)現(xiàn),阿里 EasyExcel 1.0.2 依賴的 POI 也是 3.15,所以照理說應(yīng)該是沒問題的。

如何通過IDEA快速定位和排除依賴沖突問題

但是在接口調(diào)試的時候還是出問題了,而且異常信息很奇怪,不是看一眼就能知道問題原因的并解決的。

Caused by: java.lang.AbstractMethodError: org.apache.xerces.dom.DocumentImpl.getXmlStandalone()Z
 at com.sun.org.apache.xalan.internal.xsltc.trax.DOM2TO.setDocumentInfo(DOM2TO.java:377)
 at com.sun.org.apache.xalan.internal.xsltc.trax.DOM2TO.parse(DOM2TO.java:131)
 at com.sun.org.apache.xalan.internal.xsltc.trax.DOM2TO.parse(DOM2TO.java:98)
 at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transformIdentity(TransformerImpl.java:693)
 at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:737)
 at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:351)
 at org.apache.poi.openxml4j.opc.StreamHelper.saveXmlInStream(StreamHelper.java:80)
 at org.apache.poi.openxml4j.opc.internal.marshallers.ZipPartMarshaller.marshallRelationshipPart(ZipPartMarshaller.java:181)
 at org.apache.poi.openxml4j.opc.ZipPackage.saveImpl(ZipPackage.java:560)
 at org.apache.poi.openxml4j.opc.OPCPackage.save(OPCPackage.java:1557)
 at org.apache.poi.POIXMLDocument.write(POIXMLDocument.java:248)
 at org.apache.poi.xssf.streaming.SXSSFWorkbook.write(SXSSFWorkbook.java:941)
 at com.alibaba.excel.write.ExcelBuilderImpl.finish(ExcelBuilderImpl.java:64)
 at com.alibaba.excel.ExcelWriter.finish(ExcelWriter.java:95)
 at com.pingan.haofang.creams.common.utils.ExcelUtil.writeExcel(ExcelUtil.java:71)
 ......
 ... 65 common frames omitted

提取關(guān)鍵信息,可以看到錯誤類型 java.lang.AbstractMethodError ,這個錯誤類型望名知義:抽象方法錯誤。這種類型的錯誤和我們上面說的 ClassNotFoundException 類似,很大可能就是 Jar包依賴沖突所導(dǎo)致的。

異常定位

那我們來定位下是哪個 jar 包沖突了,只需要將沖突的 jar 包排除掉,留下正確的就可以了。

我們可以看到錯誤類型是 java.lang.AbstractMethodError ,錯誤類型后面是具體的錯誤信息描述 :

org.apache.xerces.dom.DocumentImpl.getXmlStandalone()Z ,意思是在包 org.apache.xerces.dom 下的類 DocumentImpl 它的方法 getXmlStandalone() 調(diào)用出現(xiàn)了錯誤。

那么具體是誰在調(diào)用呢?我們在異常信息的緊密下一行可以看到如下這一行代碼:

at com.sun.org.apache.xalan.internal.xsltc.trax.DOM2TO.setDocumentInfo(DOM2TO.java:377)

在包路徑 com.sun.org.apache.xalan.internal.xsltc.trax 下, DOM2TO 類代碼的的第377行,有個 setDocumentInfo 方法,我們鼠標(biāo)左鍵點進去,在該行加個 Debug 斷點。

如何通過IDEA快速定位和排除依賴沖突問題

我們發(fā)現(xiàn)這個 DOM2TO 類是 JDK1.8中 rt.jar 包里面的,具體類路徑如下:

如何通過IDEA快速定位和排除依賴沖突問題

通過斷點調(diào)試得知,這個 document 對象是 DocumentImpl 實例,

如何通過IDEA快速定位和排除依賴沖突問題

這個DocumentImpl 的真實路徑也是 JDK1.8中 rt.jar 包里面的,它是 CoreDocumentImpl 的子類,CoreDocumentImpl 是接口Document 的實現(xiàn)類。

package com.sun.org.apache.xerces.internal.dom;
public class DocumentImpl
 extends CoreDocumentImpl
 implements DocumentTraversal, DocumentEvent, DocumentRange {
 ......
}
CoreDocumentImpl
package com.sun.org.apache.xerces.internal.dom;
public class CoreDocumentImpl
  extends ParentNode implements Document {
  ......
}

我們在 CoreDocumentImpl 類中第983行發(fā)現(xiàn)了 getXmlStandalone 方法。

如何通過IDEA快速定位和排除依賴沖突問題

這時報錯原因赤條條的擺在我們面前了,顯而易見,DOM2TO類中 setDocumentInfo 方法的參數(shù) Document 是屬于 JDK1.8 中 rt.jar 包下類路徑 com.sun.org.apache.xerces.internal.dom 下的實現(xiàn)類 DocumentImpl。而我們報錯的信息提示中是:

Caused by: java.lang.AbstractMethodError: org.apache.xerces.dom.DocumentImpl.getXmlStandalone()Z
這個 org.apache.xerces.dom.DocumentImpl 明顯不屬于我們 JDK1.8 的 rt.jar 包,而且也沒有 getXmlStandalone 這個方法。

所以得知,我的項目中 jar 包依賴沖突了,我們只需要排除掉 org.apache.xerces.dom.DocumentImpl 所屬的 jar 包就可以了。如何排除呢?

排除沖突

我們在 IDEA 中雙擊 Shift 鍵,輸入 DocumentImpl,得到如下結(jié)果:

如何通過IDEA快速定位和排除依賴沖突問題

可以發(fā)現(xiàn),這里有兩個 CoreDocumentImpl,一個是我們的 JDK1.8的,一個是屬于 xerce的,而且確實在依賴的 maven jar 包中發(fā)現(xiàn)了 xercesImpl-2.4.0.jar,這個 jar包就是需要排除的 jar包。

如何通過IDEA快速定位和排除依賴沖突問題

發(fā)現(xiàn)了沖突的 jar包,我全局搜索關(guān)鍵字 xerces,并沒有發(fā)現(xiàn)哪一個 pom 中有依賴的代碼,所以很可能是其他的 jar 包傳遞依賴進來的。

我們借助 IDEA 的 maven 工具,在 maven 欄右鍵項目模塊,選擇 show Dependencies 或 Ctrl + Shift + Alt + U ,這時候會展示當(dāng)前模塊的 jar 包依賴圖,如下:

如何通過IDEA快速定位和排除依賴沖突問題

雖然這里展示了很多沖突的jar包,其中紅線連接的就是沖突的jar 包,但是我們 Ctrl + F 查詢 xerces 還是沒有結(jié)果。

所以我們需要額外的方式來解決,這時我想到了 IDEA 有個插件 Maven Helper ,具體的插件下載可以參考前面的內(nèi)容,下載好插件后,我們打開 pom.xml 文件,在pom.xml 文件的左下方有個 Dependency Analyzer ,我們點擊之后顯示如下:

如何通過IDEA快速定位和排除依賴沖突問題

  • Conflicts :展示所有沖突。

  • All Dependencies as List :以列表的方式展示所有依賴。

  • All Dependencies as Tree :以樹形的方式展示所有依賴。

我們輸入 xerces,選擇以樹形展示所有依賴,得到如下的信息顯示。

如何通過IDEA快速定位和排除依賴沖突問題

清晰明了,原來這個罪魁禍?zhǔn)资潜?file-web-sdk 帶進來的,我們右鍵選擇 Jump To Source 或者 F4 定位到這個 jar 在 pom.xml 的依賴引入位置,如下圖所示,我們通過 exclusion 標(biāo)簽排除 xercesImpl 的引入即可。

<dependency>
 <groupId>com.xx.xx.gov.fileservice</groupId>
 <artifactId>file-web-sdk</artifactId>
 <exclusions>
  <exclusion>
   <groupId>xerces</groupId>
   <artifactId>xercesImpl</artifactId>
  </exclusion>
 </exclusions>
</dependency>

再次啟動項目,測試接口發(fā)現(xiàn)功能正常了,整個排查過程也就結(jié)束了,IDEA的功能還是很強大的。

以上是“如何通過IDEA快速定位和排除依賴沖突問題”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI