以下是一些關于Maven的經典面試題以及它們的答案:?
1、什么是Maven?
Maven是一個項目管理工具,用于構建、管理、發布Java項目。
2、為什么要使用Maven而不是手動管理項目依賴?
Maven提供了依賴管理、統一的構建、打包、文檔生成等功能,能幫助開發者減少手動管理項目的繁瑣和錯誤,提高了項目的可維護性和可靠性。
3、Maven的生命周期是什么?以及各個階段的含義是什么?
Maven的生命周期是指Clean、Build、Site三個階段,每個階段又有多個子階段。Clean階段負責清理項目;Build階段負責編譯、測試、打包、安裝和部署項目;Site階段負責生成項目的站點文檔。Maven的構建生命周期按照Clean、Build、Site的順序執行,每個階段都有特定的Maven插件去執行具體的構建任務,Maven提供了許多默認的插件,同時也允許開發者自定義和擴展插件,以滿足項目的特定需求。
- Clean階段。這個周期主要用于清理項目,刪除上一次構建生成的target文件。它的主要階段包括pre-clean、clean和post-clean。pre-clean執行一些需要在clean之前完成的工作,clean移除所有上一次構建生成的文件,post-clean執行一些需要在clean之后立刻完成的工作。
- Default階段。這個周期是構建和發布項目的核心部分,包括編譯、測試、打包、安裝和部署等步驟。它的主要階段包括validate、compile、test、package、verify、install和deploy。validate驗證項目結構和配置文件是否是完整的正確的可用的,compile將源代碼編譯為可執行的字節碼,test使用單元測試框架測試編譯后的源代碼,package獲取已編譯的代碼,并將其打包為可發布的格式,如JAR,verify驗證包是否滿足Maven的構建規范,install將包安裝到本地倉庫,以供其他項目使用,deploy將包部署到遠程倉庫。
- Site階段。這個周期主要用于生成和發布項目的站點文檔。它的主要階段包括pre-site、site、post-site和site-deploy。pre-site完成站點文檔生成前的準備工作,比如創建目錄結構、復制靜態內容等,site生成項目的站點文檔(在這個階段,Maven會根據項目配置的站點文檔生成插件,生成HTML、XML、PDF等格式的文檔),生成的文檔可以包含項目信息:如描述、開發者、許可證等,Javadocs:描述項目中的類、接口和方法,報告:如單元測試報告、代碼覆蓋率報告(代碼覆蓋率表示測試用例執行時覆蓋的代碼量與總代碼量的比例)、代碼質量報告等,依賴列表:列出項目使用的依賴,模塊關系:如果項目是多模塊的,這部分將展示各模塊之間的關系,post-site用于完成生成站點文檔后的附加工作,site-deploy將生成的站點文檔部署到特定的服務器上。
4、Maven的依賴管理是什么?如何在POM文件中定義依賴?
Maven的依賴管理是指通過POM文件來管理項目的外部依賴庫。可以在POM文件的dependencies標簽下,使用groupId、artifactId和version來定義依賴。
- groupId:這是項目組的標識。它在一個組織或者項目中通常是唯一的。
- artifactId:這是項目的標識,通常是工程的名稱。它在一個項目組(group)下是唯一的。
- version:這是項目的版本號,用來區分同一個artifact的不同版本。
5、Maven的倉庫是什么?有哪些倉庫類型?
Maven的倉庫是存儲項目依賴庫的地方。類型包括本地倉庫、遠程倉庫、中央倉庫、私有倉庫、快照倉庫(Snapshot版本)和發行倉庫(Release 版本)等。
6、如何發布項目到私有倉庫?
可以在Maven的settings.xml文件中配置私有倉庫的URL和認證信息,然后使用mvn deploy命令來發布項目到私有倉庫。
7、如何跳過單元測試的執行?
可以使用-DskipTests=true參數來跳過單元測試的執行,如mvn clean install -DskipTests=true。
8、如何指定Maven使用的JDK版本?
第一種(最省事):在maven的conf目錄下修改settings.xml文件,在profiles中加入以下配置,設置maven.compiler.source和maven.compiler.target來指定使用的JDK版本。
<profile> <id>jdk-1.8</id> <activation> <activeByDefault>true</activeByDefault> <jdk>1.8</jdk> </activation> <properties> <maven.compiler.source>1.8</maven.compiler.source> <maven.compiler.target>1.8</maven.compiler.target> <maven.compiler.compilerVersion>1.8</maven.compiler.compilerVersion> </properties>
</profile>
第二種:在pom.xml中添加配置。
<properties><maven.compiler.source>1.8</maven.compiler.source><maven.compiler.target>1.8</maven.compiler.target><maven.compiler.encoding>UTF-8</maven.compiler.encoding><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
</properties>
第三種:在pom.xml中添加配置,通過maven的插件實現。
<build><plugins><!-- 設置編譯版本 --><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><configuration><source>1.8</source><target>1.8</target><encoding>UTF-8</encoding></configuration></plugin>
</plugins>
9、描述一下Maven中的傳遞性依賴?如何排除傳遞性依賴?
Maven中的傳遞性依賴是指通過直接依賴間接依賴了其它的依賴庫。Maven會自動解析和加載這些傳遞性依賴。可以在POM文件的依賴聲明中使用exclusions標簽來排除傳遞性依賴。
10、如何創建一個Maven項目?
可以通過idea提供的菜單面板創建maven項目,也可以通過命令mvn archetype:generate創建。?
mvn archetype:generate
-DarchetypeGroupId=groupId
-DarchetypeArtifactId=artifactId
-DarchetypeVersion=version
-DgroupId=groupId
-DartifactId=artifactId
-Dversion=version
11、如何更新Maven依賴的版本?
可以使用mvn versions:update-parent或mvn versions:use-latest-versions命令將項目中的依賴項更新到最新版本。如要顯示項目中的依賴項是否有可用的更新版本,有命令mvn versions:display-dependency-updates
?。
12、Maven中的optional和scope以及type作用?
optional
為true表示將依賴項設為可選,這樣依賴該項目的項目,就不會引入該依賴,減少了依賴傳遞,避免依賴沖突,減小了jar和war體積,默認值為 false。?在某些情況下,項目可能依賴于兩個或多個包含相互沖突的依賴項的庫,通過將這些沖突的依賴項設置為可選,我們可以讓項目顯式地選擇使用哪個版本,從而消除沖突。?
常見的有?spring-boot-devtools 熱部署,junit 單元測試,lombok等生產環境下不需要的依賴包。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-devtools</artifactId><optional>true</optional> </dependency> <dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><optional>true</optional> </dependency> <dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional> </dependency>
scope
scope 元素主要用來控制依賴的使用范圍。
<scope>import</scope>的import只能在dependencyManagement中使用,且type必須為pom:
type
type:指明引入的依賴的類型(jar、war、pom等),默認jar。用到<scope>import</scope>時必須要聲明<type>pom</type>。而<scope>import</scope>的import只能在dependencyManagement中用,所以<type>pom</type>常用在dependencyManagement標簽中。
dependencyManagement標簽用法會在本文章標題13中講到。
13、Maven多模塊項目中的繼承和聚合有什么作用?
模塊的“繼承”
子 pom 通過繼承父 pom 來復用父 pom的配置和依賴。
可繼承的標簽
可繼承的標簽太多了,不一一例舉,這里就列一些我們在開發中常用的可繼承的標簽:
groupId、version:“項目所屬組織”和“版本”可以繼承(artifactId “模塊名”不能繼承)。
dependencies:“依賴”可以繼承。
denpendencyManagement:“依賴管理”配置可以繼承。
我們往往會把子模塊中使用到的依賴以及版本號等,抽取到父模塊中由子模塊直接繼承。
一般是用在最頂層的父pom,子模塊繼承之后鎖定版本,能讓所有的子模塊中引用一個依賴而不用顯式的列出版本號,Maven會沿著父子層向上走,直到找到一個擁有dependencyManagement元素的模塊,然后他就會使用這個dependencyManagement元素中指定的版本號。這么做的好處就是,如果有多個子模塊都引用同樣的依賴,則可以避免在每個使用的子模塊中的聲明一個版本號。只需要在頂層父模塊里更新,而不需要對子模塊逐一進行修改,另外如果某個子項目中需要單獨使用另外的版本,只需要在子模塊中單獨聲明version即可。dependencyManagement里面只是聲明和限定依賴版本的,并未實現依賴的引入,因此子項目仍需要顯式地聲明所要引入的依賴。如果不在子項目中聲明依賴,是不會從父項目中繼承下來的,只有在子項目中引入了此依賴并且沒有指定具體的版本號,才會從父項目中繼承該項,并且version和scope都讀取自父pom。如果子項目中定義了版本號,則會使用子項目中定義的版本號。
properties:“自定義屬性”(類似于定義一個變量)可以繼承。
repositories:倉庫配置。
pom.xml
文件中的?repositories
?元素用于定義Maven構建系統在構建當前項目時從哪里獲取依賴項。<repositories><repository><id>central</id><url>https://repo.maven.apache.org/maven2</url></repository><!-- 其他存儲庫的定義 --> </repositories>在這個例子中,repositories?包含了一個名為 “central” 的遠程存儲庫, 該存儲庫的URL是Maven中央存儲庫的地址。 項目構建時,Maven將會在這個存儲庫中查找項目所需的依賴項。
將項目依賴的倉庫信息直接定義在 pom.xml 中有一些優勢:項目的構建不依賴于全局配置,可在不同環境中順利構建,降低了對外部環境的依賴,使得項目更具可移植性 。
在實際的依賴項解析過程中,Maven按照以下順序查找依賴項:本地倉庫-->pom.xml 中的 repositories 標簽配置的遠程倉庫-->?Maven 配置文件 settings.xml 中的遠程倉庫配置-->
Maven內置默認倉庫,例如 Maven Central Repository。在這個過程中,一旦找到符合條件的依賴項,Maven就會停止搜索,避免不必要的網絡請求。
distributionManagement:項目deploy的私服倉庫配置。
build:插件、插件管理、源碼輸出位置等配置。
列舉幾個配置案例。
默認情況下,如果沒有指定resources,自動會將classpath下的src/main/java下的.class文件和src/main/resources下的.xml文件放到target的classes文件夾下的package下的文件夾里。但如果設定了resources,那么默認的就會失效,就會以設置的為準。
pom.xml中的build標簽的resources標簽_pom build resources-CSDN博客
<resource><directory>src/main/resources</directory> </resource>
resource表示讀取該目錄的所有文件。
<resource><directory>src/main/resources</directory><filtering>true</filtering> </resource>
filtering是否開啟標簽替換,無此標簽表示不開啟替換。若讀取的文件中有類似${key}這樣的配置,<filtering>true</filtering>就會從pom.xml中profiles的配置里面取。
<resource><directory>src/main/resources</directory><targetPath>META-INF/plexus</targetPath> </resource>
targetPath用于指定讀取資源到target的那個目錄下,如下圖,不指定默認為target/classes
<resource><directory>src/main/resources</directory><includes><include>config/dubboSource/*.xml</include></includes> </resource>
includes表示僅讀取directory文件夾下includes中指定的文件或文件夾的內容,即in的意思,如下圖展示
<resource><directory>src/main/resources</directory><excludes><exclude>config/dubboSource/*.xml</exclude></excludes> </resource>
excludes表示讀取directory文件夾下但排除excludes中指定的文件或文件夾的所有其他內容,即not in的意思。
<testResources><testResource><directory>src/test/resources</directory><filtering>true</filtering></testResource> </testResources>
testResources這個模塊用來配置測試資源元素,其內容定義與resources類似。
子POM中引入父POM
父pom:
子pom引入父pom:
relativePath
?指的是父?pom.xml
?所在的相對路徑,默認值是?../pom.xml
,如下圖:這也是 Maven 建議的目錄組織形式,如果想將 parent模塊?與 child 模塊放到同一級目錄,則需要修改為:
<relativePath>../parent模塊名/pom.xml</relativePath>
,如下圖:
模塊的“聚合”
聚合(aggregation)用于按順序批量構建子模塊。
聚合的注意事項
在這個項目中,demo-a 依賴 demo-b, demo-b 依賴 demo-c , 在這種情況下,我們需要先 install c ,再 install b ,最后再構建 a,執行起來非常麻煩,為了處理這個問題,Maven 引入了聚合機制,可以將這三個模塊聚合在一起,一次性完成構建。
聚合的語法也非常簡單,只需要在父模塊的?pom.xml
?中添加?<modules>
?即可:
<modules><module>demo-a</module><module>demo-b</module><module>demo-c</module>
</modules>
<!-- 這里需要注意的是,<module> 中填寫的并不是 artifactId,
而是需要被聚合的模塊的 文件目錄 的相對路徑! -->
反應堆(reactor)
在一個多模塊的Maven項目中,反應堆(Reactor)是指所有模塊組成的一個構建結構,對于單模塊的項目,反應堆就是該模塊本身。我們使用上面聚合的項目來做一次構建:
mvn clean install
注意上面的執行順序,先構建的是配置了聚合關系的父模塊,然后才是子模塊,子模塊構建的時候會檢查有沒有依賴的模塊,如果有就先構建被依賴的模塊。所以這里按照被依賴的順序,由底層向上層進行構建。因此,我們的模塊之間不能出現循環依賴的情況,假如在demo-c中又引入demo-a的依賴,此時構建就會報錯。
The projects in the reactor contain a cyclic reference: ......
在實際開發中,有時項目比較龐大,如果每次都完整的進行構建,耗時會很長。此時我們可以在構建指令上加上一些參數,以便于我們選擇性的構建我們需要的項目,通過mvn -h
查看。?
usage: mvn [options] [<goal(s)>] [<phase(s)>]Options:-am,--also-make If project list is specified, alsobuild projects required by thelist-amd,--also-make-dependents If project list is specified, alsobuild projects that depend onprojects on the list-B,--batch-mode Run in non-interactive (batch)mode (disables output color)-b,--builder <arg> The id of the build strategy touse-C,--strict-checksums Fail the build if checksums don'tmatch-c,--lax-checksums Warn if checksums don't match-cpu,--check-plugin-updates Ineffective, only kept forbackward compatibility-pl,--projects <arg> Comma-delimited list of specifiedreactor projects to build insteadof all projects. A project can bespecified by [groupId]:artifactIdor by its relative path
上面的-pl
和-am
是比較常用的參數:
pl
:指定構建某一個項目am
:構建項目的同時,構建它依賴的項目
假如我想指定構建demo-b
及其依賴的模塊demo-c
,則可以用下面的指令:
mvn clean install -pl demo-b -am
14、Maven的插件是什么?如何配置插件?
插件是用于擴展Maven功能的工具,可以在項目的pom.xml文件中的build標簽的子標簽plugins中配置插件的參數來使用插件。
15、什么是Snapshot、Release版本?
Snapshot版本是指開發中的不穩定版本,用于表示項目的最新開發狀態。一般來說Snapshot版本代表正在開發中的版本,而Release版本代表比較穩定的發布版本。
16、如何在Maven中配置插件?
在POM文件的build節點下使用插件塊來配置插件。可以指定插件的groupId、artifactId、version,以及插件執行的目標等。
Maven中的profile是什么?如何使用profile?
Profile是一種Maven的特性,用于根據不同的環境配置來激活不同的構建選項。可以在POM文件中使用profile標簽定義profile,并使用activation標簽來指定profile的激活條件。
如何在Maven中跳過特定的構建階段?
可以使用命令行參數或配置來跳過特定的構建階段。例如,使用-Dmaven.test.skip=true參數可以跳過測試階段。
Maven的聚合和繼承有什么區別?什么時候使用聚合,什么時候使用繼承?
聚合(aggregation)是將多個相互獨立的模塊組合成一個大項目。繼承(inheritance)是使用父POM中的配置和依賴。聚合用于管理多模塊項目,繼承用于重用配置和依賴。
Maven中的插件管理是什么?如何在POM文件中配置插件管理?
插件管理是用于集中管理項目中使用的插件的配置。可以在POM文件的build節點下使用pluginManagement標簽來配置插件管理,然后在plugins標簽下配置具體的插件。
Maven中的依賴范圍有哪些?
Maven的依賴范圍包括:compile、provided、runtime、test和system。
Maven中的依賴沖突是什么?如何解決依賴沖突?
依賴沖突是指項目中多個依賴庫引用了同一個依賴的不同版本,可能導致運行時的沖突。可以使用Maven的dependencyManagement機制來解決依賴沖突。
如何創建Maven插件?
可以使用Maven的插件骨架生成器(Maven Plugin Archetype Generator)來創建Maven插件的初始模板,并在模板基礎上進行開發。?