談?wù)?Maven 輕松重構(gòu)項(xiàng)目
現(xiàn)在是微服務(wù)盛行時(shí)代,說不準(zhǔn)哪一天領(lǐng)導(dǎo)就會(huì)讓你對一個(gè)大項(xiàng)目進(jìn)行重構(gòu)。大項(xiàng)目的痛點(diǎn):編譯慢、發(fā)布繁瑣等。就像下面這張圖:
真的不敢動(dòng)呀,一不小心就坍塌了。
比如說我們用戶系統(tǒng),我們可以這么重構(gòu)(這里只是舉例,每個(gè)項(xiàng)目拆分目的可能不同)。
user-system拆分成:
- user-web
- user-service
- user-dao
- user-common
我們對其拆分后很有可能存在多個(gè)子項(xiàng)目中同時(shí)依賴某個(gè)jar包,如果使用不當(dāng)可能會(huì)導(dǎo)致每個(gè)模塊使用的版本不一樣,所以想想能不能有什么方式來解決這類問題呢?
答案:能。但是得先了解兩個(gè)概念:聚合和繼承。
聚合
所謂的聚合就是我們?nèi)绻胍淮涡詷?gòu)建多個(gè)項(xiàng)目模塊,那我們就需要把這些項(xiàng)目模塊進(jìn)行聚合。
配置模板
- <modules>
- <module>項(xiàng)目模塊一</module>
- <module>項(xiàng)目模塊二</module>
- <module>項(xiàng)目模塊三</module>
- <module>項(xiàng)目模塊...</module>
- </modules>
使用方式
比如我們對user-web、user-service、user-dao、user-common四個(gè)項(xiàng)目模塊進(jìn)行聚合。
- <modules>
- <module>user-web</module>
- <module>user-service</module>
- <module>user-dao</module>
- <module>user-common</module>
- </modules>
注意:其中module的路徑為相對路徑。
這樣就解決了我們項(xiàng)目模塊的聚合,我們再來看看maven是如何來解決由于重復(fù)依賴導(dǎo)致版本不一樣的問題。
繼承
繼承為了消除重復(fù),我們把很多相同的配置提取出來,例如:grouptId,version等 。
繼承的配置
- <parent>
- <artifactId>maven-demo</artifactId>
- <groupId>com.tian</groupId>
- <version>1.0-SNAPSHOT</version>
- </parent>
繼承代碼中定義屬性
繼承代碼過程中,可以定義屬性,例如:
- <properties>
- <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
- <junit.version>4.13.1</junit.version>
- <maven.version>0.0.1-SNAPSHOT</maven.version>
- </properties>
訪問屬性的方式為${junit.version},例如:
- <?xml version="1.0" encoding="UTF-8"?>
- <project xmlns="http://maven.apache.org/POM/4.0.0"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
- <parent>
- <artifactId>maven-demo</artifactId>
- <groupId>com.tian</groupId>
- <version>1.0-SNAPSHOT</version>
- </parent>
- <modelVersion>4.0.0</modelVersion>
- <artifactId>user-web</artifactId>
- <dependencies>
- <dependency>
- <groupId>junit</groupId>
- <artifactId>junit</artifactId>
- <!-- 使用了parent中 junit.version -->
- <version>${junit.version}</version>
- </dependency>
- </dependencies>
- </project>
這樣我們就可以使用到上面定義的屬性junit.version=4.13.1。
父模塊用dependencyManagement進(jìn)行管理
由于項(xiàng)目模塊較多,所以我們又抽象了一層,抽出一個(gè)parent來管理子項(xiàng)目的公共的依賴。為了項(xiàng)目的正確運(yùn)行,必須讓所有的子項(xiàng)目使用依賴項(xiàng)的統(tǒng)一版本,必須確保應(yīng)用的各個(gè)項(xiàng)目的依賴項(xiàng)和版本一致,才能保證測試的和發(fā)布的是相同的結(jié)果。
- <dependencyManagement>
- <dependencies>
- <dependency>
- <groupId>junit</groupId>
- <artifactId>junit</artifactId>
- <version>${junit.version}</version>
- <scope>test</scope>
- </dependency>
- </dependencies>
- </dependencyManagement>
這樣的好處是子模塊可以有選擇性的繼承,而不需要全部繼承。
這樣做的好處:統(tǒng)一管理項(xiàng)目的版本號,確保應(yīng)用的各個(gè)項(xiàng)目的依賴和版本一致,才能保證測試的和發(fā)布的是相同的成果,因此,在頂層pom中定義共同的依賴關(guān)系。同時(shí)可以避免在每個(gè)使用的子項(xiàng)目中都聲明一個(gè)版本號,這樣想升級或者切換到另一個(gè)版本時(shí),只需要在父類容器里更新,不需要任何一個(gè)子項(xiàng)目的修改;如果某個(gè)子項(xiàng)目需要另外一個(gè)版本號時(shí),只需要在dependencies中聲明一個(gè)版本號即可。子類就會(huì)使用子類聲明的版本號,不繼承于父類版本號。
聚合和繼承的關(guān)系
在前面我們構(gòu)建多模塊項(xiàng)目中,關(guān)系如下
mavendemo就是user-web、user-service等幾個(gè)模塊的父項(xiàng)目。
父項(xiàng)目主要是用來配置一些公共的配置,其它三個(gè)項(xiàng)目再通過繼承的方式擁有父項(xiàng)目中的配置,首先配置父項(xiàng)目的pom.xml,添加對項(xiàng)目的user-web、user-service、user-dao、user-common這四個(gè)模塊進(jìn)行聚合以及jar包依賴。
相對于dependencyManagement,所有生命在dependencies里的依賴都會(huì)自動(dòng)引入,并默認(rèn)被所有的子項(xiàng)目繼承。
面試題:dependencyManagement和dependencies的區(qū)別
- dependencyManagement里只是聲明依賴,并不實(shí)現(xiàn)引入,因此子項(xiàng)目需要顯示的聲明需要用的依賴。如果不在子項(xiàng)目中聲明依賴,是不會(huì)從父項(xiàng)目中繼承下來的;只有在子項(xiàng)目中寫了該依賴項(xiàng),并且沒有指定具體版本,才會(huì)從父項(xiàng)目中繼承該項(xiàng),并且version和scope都讀取自父pom;另外如果子項(xiàng)目中指定了版本號,那么會(huì)使用子項(xiàng)目中指定的jar版本。
- dependencies即使在子項(xiàng)目中不寫該依賴項(xiàng),那么子項(xiàng)目仍然會(huì)從父項(xiàng)目中繼承該依賴項(xiàng)(全部繼承)。
IDEA中配置
Maven在使用IDEA開發(fā)時(shí),如何將Maven配置呢?
打開IDEA,在File--->Settings。輸入欄搜索:maven
配置我們之前安裝好的maven目錄、maven下面的setting.xml以及本地倉庫目錄。然后Apply--->OK。
我們項(xiàng)目對某個(gè)項(xiàng)目進(jìn)行clean 等操作的時(shí)候,就可以直接雙擊Lifcycle下面的。
這樣我們就把Maven集成到我們的IDEA中來了。
編譯運(yùn)行項(xiàng)目
我們可以在父項(xiàng)目中對所有子項(xiàng)目進(jìn)行編譯、打包等。我們就來對所有子模塊進(jìn)行打包。
然后在對應(yīng)子項(xiàng)目中可以找到target目錄和對應(yīng)的jar包。
也可單獨(dú)對某個(gè)子項(xiàng)目進(jìn)行打包等操作。比如我們對user-common進(jìn)行clean。
于是,此時(shí)的user-common下的target已經(jīng)被刪掉,其他子項(xiàng)目中并沒有收到影響。
IDEA中如何創(chuàng)建多項(xiàng)目模塊
先創(chuàng)建一個(gè)user-parent的maven項(xiàng)目:
然后把src目錄刪掉,創(chuàng)建子項(xiàng)目:
選中user-parent項(xiàng)目,右鍵,選中New-->Module
來到New Module界面,選擇Maven,選好自己的本地安裝的JDK。
next。填寫子項(xiàng)目名稱,我們這里創(chuàng)建一個(gè)user-web的子項(xiàng)目:
注意這里user-web,默認(rèn)是userweb,需要我們手工在兩個(gè)單詞之間添加橫線:
點(diǎn)擊Finish:
然后在看看user-parent的pom中的內(nèi)容:
已經(jīng)添加了modules,同時(shí)把我們的user-web添加進(jìn)去了。
同樣方法,我們就可以創(chuàng)建更多字項(xiàng)目了:
再看看user-parent中的pom.xml文件內(nèi)容:
到這里,我們就輕輕松松的構(gòu)建了我們這個(gè)大項(xiàng)目user-parent,該項(xiàng)目包括了user-web,user-service,user-dao,-user-commmon子項(xiàng)目。
若是新建多模塊項(xiàng)目,安裝上面的步驟就搞定了。
若是大項(xiàng)目拆分,我們就可以使用上面的方式先創(chuàng)建好自己的多模塊項(xiàng)目,然后對應(yīng)把大項(xiàng)目里的內(nèi)容拷貝到對應(yīng)模塊中(需要注意此時(shí)子模塊直接肯定會(huì)有相互依賴的)。
總結(jié)
通過繼承,我們可以定義項(xiàng)目的父類項(xiàng)目,用于繼承父項(xiàng)目的依賴、插件、屬性等信息。聚合就是我們?nèi)绻胍淮涡詷?gòu)建多個(gè)項(xiàng)目模塊,那我們就需要對這些項(xiàng)目模塊進(jìn)行聚合。
構(gòu)建多模塊項(xiàng)目,在IDEA中使用創(chuàng)建Module的方式很輕松就可以創(chuàng)建了。在父項(xiàng)目中使用來管理子模塊的依賴相關(guān)依賴。
對大項(xiàng)目進(jìn)行重構(gòu)時(shí),只需要把各模塊拆分出來的代碼拷貝到對應(yīng)子模塊就可以了。
本文轉(zhuǎn)載自微信公眾號「 Java后端技術(shù)全?!梗梢酝ㄟ^以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系 Java后端技術(shù)全棧公眾號。