前言 Maven,發(fā)音是[`meivin],專家的意思。它是一個很好的項目管理工具,很早就進入了我的必備工具行列,但是這次為了把ABPM項目 完全遷移并應用maven,所以對maven進行了一些深入的學習。寫這個學習筆記的目的,一個是為了自己備忘,二則希望能夠為其他人
Maven,發(fā)音是[`meivin],"專家"的意思。它是一個很好的項目管理工具,很早就進入了我的必備工具行列,但是這次為了把ABPM項目 完全遷移并應用maven,所以對maven進行了一些深入的學習。寫這個學習筆記的目的,一個是為了自己備忘,二則希望能夠為其他人學習使用maven 縮短一些時間。
首先我把maven的概念快速的梳理一下,讓我們快速地建立起一個比較精確的maven應用場景。
讀書時候要先限定范圍,避免一些有害的遐想。要說maven不是什么,我們可以從如下幾個要點來展開
maven將自己定位為一個項目管理工具。它負責管理項目開發(fā)過程中的幾乎所有的東西:
maven把項目的構建劃分為不同的生命周期(lifecycle),在我看來,劃分的已經是非常仔細了,大家可以參考這里 。粗略一點的話,它這個過程(phase)包括:編譯、測試、打包、集成測試、驗證、部署。maven中所有的執(zhí)行動作(goal)都需要指明自己在這個過程中的執(zhí)行位置,然后maven執(zhí)行的時候,就依照過程的發(fā)展依次調用這些goal進行各種處理。
這個也是maven的一個基本調度機制。一般來說,位置稍后的過程都會依賴于之前的過程。當然,maven同樣提供了配置文件,可以依照用戶要求,跳過某些階段。
所謂的"約定優(yōu)于配置",在maven中并不是完全不可以修改的,他們只是一些配置的默認值而已。但是使用者除非必要,并不需要去修改那些約定內容。maven默認的文件存放結構如下:
每一個階段的任務都知道怎么正確完成自己的工作,比如compile任務就知道從src/main/java下編譯所有的java文件,并把它的輸出class文件存放到target/classes中。
對maven來說,采用"約定優(yōu)于配置"的策略可以減少修改配置的工作量,也可以降低學習成本,更重要的是,給項目引入了統(tǒng)一的規(guī)范。
maven使用如下幾個要素來唯一定位某一個輸出物: groupId:artifactId:packaging:version 。比如org.springframework:spring:2.5 。每個部分的解釋如下:
maven有自己的版本規(guī)范,一般是如下定義
maven在版本管理時候可以使用幾個特殊的字符串 SNAPSHOT ,LATEST ,RELEASE 。比如"1.0-SNAPSHOT"。各個部分的含義和處理邏輯如下說明:
maven把整個maven管理的項目分為幾個部分,一個部分是源代碼,包括源代碼本身、相關的各種資源,一個部分則是單元測試用例,另外一部分則是各種maven的插件。對于這幾個部分,maven可以獨立管理他們,包括各種外部依賴關系。
依賴管理一般是最吸引人使用maven的功能特性了,這個特性讓開發(fā)者只需要關注代碼的直接依賴,比如我們用了spring,就加入spring依賴說明就可以了,至于spring自己還依賴哪些外部的東西,maven幫我們搞定。
任意一個外部依賴說明包含如下幾個要素:groupId, artifactId, version, scope, type, optional。其中前3個是必須的,各自含義如下:
maven認為,程序對外部的依賴會隨著程序的所處階段和應用場景而變化,所以maven中的依賴關系有作用域(scope)的限制。在maven中,scope包含如下的取值:
另外,代碼有代碼自己的依賴,各個maven使用的插件也可以有自己的依賴關系。依賴也可以是可選的,比如我們代碼中沒有任何cache依賴,但是hibernate可能要配置cache,所以該cache的依賴就是可選的。
maven的多項目管理也是非常強大的。一般來說,maven要求同一個工程的所有子項目都放置到同一個目錄下,每一個子目錄代表一個項目,比如
按照這種格式存放,就是繼承方式,所有具體子項目的pom.xml都會繼承總項目pom的內容,取值為子項目pom內容優(yōu)先。
要設置繼承方式,首先要在總項目的pom中加入如下配置
Xml代碼
其次在每個子項目中加入
org.sonatype.mavenbook.ch06 simple-parent1.0
即可。
當然,繼承不是唯一的配置文件共用方式,maven還支持引用方式。引用pom的方式更簡單,在依賴中加入一個type為pom的依賴即可。
Xml代碼
用戶可以在maven中定義一些屬性,然后在其他地方用${xxx}進行引用。比如:
Xml代碼
maven提供了三個隱式的變量,用來訪問系統(tǒng)環(huán)境變量、POM信息和maven的settings:
profile是maven的一個重要特性,它可以讓maven能夠自動適應外部的環(huán)境變化,比如同一個項目,在linux下編譯linux的版 本,在win下編譯win的版本等。一個項目可以設置多個profile,也可以在同一時間設置多個profile被激活(active)的。自動激活的 profile的條件可以是各種各樣的設定條件,組合放置在activation節(jié)點中,也可以通過命令行直接指定。profile包含的其他配置內容可 以覆蓋掉pom定義的相應值。如果認為profile設置比較復雜,可以將所有的profiles內容移動到專門的 profiles.xml 文件中,不過記得和pom.xml放在一起。
activation節(jié)點中的激活條件中常見的有如下幾個:
maven的操作有兩種方式,一種是通過mvn命令行命令,一種是使用maven的eclipse插件。因為使用eclipse的maven插件操作起來比較容易,這里就只介紹使用mvn命令行的操作。
maven的主執(zhí)行程序為mvn.bat,linux下為mvn.sh,這兩個程序都很簡單,它們的共同用途就是收集一些參數(shù),然后用 java.exe來運行maven的Main函數(shù)。maven同樣需要有配置文件,名字叫做settings.xml,它放在兩個地方,一個是maven 安裝目錄的conf目錄下,對所有使用該maven的用戶都起作用,我們稱為主配置文件,另外一個放在 %USERPROFILE%/.m2/settings.xml下,我們成為用戶配置文件,只對當前用戶有效,且可以覆蓋主配置文件的參數(shù)內容。還有就是 項目級別的配置信息了,它存放在每一個maven管理的項目目錄下,叫pom.xml,主要用于配置項目相關的一些內容,當然,如果有必要,用戶也可以在 pom中寫一些配置,覆蓋住配置文件和用戶配置文件的設置參數(shù)內容。
一般來說,settings文件配置的是比如repository庫路徑之類的全局信息,具體可以參考官方網站的文章 。
要創(chuàng)建一個新的maven工程,我們需要給我們的工程指定幾個必要的要素,就是maven產品坐標的幾個要素:groupId, artifactId,如果愿意,你也可以指定version和package名稱。我們先看一個簡單的創(chuàng)建命令:
d:\work\temp>mvn archetype:create -DgroupId=com.abc -DartifactId=product1 -DarchetypeArtifactId=maven-archetype-webapp
首先看這里的命令行參數(shù)的傳遞結構,怪異的 -D參數(shù)=值 的方式是 java.exe 要求的方式。這個命令創(chuàng)建一個web工程,目錄結構是一個標準的maven結構,如下:
D:. └─mywebapp │ pom.xml │ └─src └─main ├─resources └─webapp │ index.jsp │ └─WEB-INF web.xml
大家要注意,這里目錄結構的布局實際上是由參數(shù) archetypeArtifactId 來決定的,因為這里傳入的是 maven-archetype-webapp如果我們傳入其他的就會創(chuàng)建不同的結構,默認值為 maven-archetype-quickstart ,有興趣的讀者可以參考更詳細的列表 ,我把部分常用的列表在這里:
Artifact Group Version Repository Descriptionmaven-archetype-j2ee-simple | org.apache.maven.archetypes | A simple J2EE Java application | ||
maven-archetype-marmalade-mojo | org.apache.maven.archetypes | A Maven plugin development project using marmalade | ||
maven-archetype-plugin | org.apache.maven.archetypes | A Maven Java plugin development project | ||
maven-archetype-portlet | org.apache.maven.archetypes | A simple portlet application | ||
maven-archetype-profiles | org.apache.maven.archetypes | |||
maven-archetype-quickstart | org.apache.maven.archetypes | |||
maven-archetype-simple | org.apache.maven.archetypes | |||
maven-archetype-site-simple | org.apache.maven.archetypes | A simple site generation project | ||
maven-archetype-site | org.apache.maven.archetypes | A more complex site project | ||
maven-archetype-webapp | org.apache.maven.archetypes | A simple Java web application | ||
maven-archetype-har | net.sf.maven-har | 0.9 | Hibernate Archive | |
maven-archetype-sar | net.sf.maven-sar | 0.9 | JBoss Service Archive |
大家可以參考更詳細的 archetype:create 幫助 和 archtype參考信息 。
多項目管理是maven的主要特色之一,對于一個大型工程,用maven來管理他們之間復雜的依賴關系,是再好不過了。maven的項目配置之間的關系有兩種:繼承關系和引用關系。
maven默認根據(jù)目錄結構來設定pom的繼承關系,即下級目錄的pom默認繼承上級目錄的pom。要設定兩者之間的關系很簡單,上級pom如下設置:
Xml代碼
要記住的是,這里的module是目錄名,不是子工程的artifactId。子工程如下設置:
Xml代碼
這樣兩者就相互關聯(lián)起來了,繼承關系就設定完畢,所有父工程的配置內容都會自動在子工程中生效,除非子工程有相同的配置覆蓋。如果你不喜歡層層遞進的目錄結構來實現(xiàn)繼承,也可以在parent中加入
引用關系是另外一種復用的方式,maven中配置引用關系也很簡單,加入一個 type 為 pom 的依賴即可。
Xml代碼
但是無論是父項目還是引用項目,這些工程都必須用 mvn install 或者 mvn deploy 安裝到本地庫才行,否則會報告依賴沒有找到,eclipse編譯時候也會出錯。
需要特別提出的是復用過程中,父項目的pom中可以定義 dependencyManagement 節(jié)點,其中存放依賴關系,但是這個依賴關系只是定義,不會真的產生效果,如果子項目想要使用這個依賴關系,可以在本身的 dependency 中添加一個簡化的引用
Xml代碼
這種方法可以避免版本號滿天飛的情況。
在maven中一般都會用到安裝庫文件的功能,一則是我們常用的hibernate要使用jmx庫,但是因為sun的license限制,所以無法將其直接包含在repository中。所以我們使用mvn命令把jar安裝到我們本地的repository中
mvn install:install-file -DgroupId=com.sun.jdmk -DartifactId=jmxtools -Dversion=1.2.1 -Dpackaging=jar -Dfile=/path/to/file
如果我們想把它安裝到公司的repository中,需要使用命令
mvn deploy:deploy-file -DgroupId=com.sun.jdmk -DartifactId=jmxtools -Dversion=1.2.1 -Dpackaging=jar -Dfile=/path/to/file -Durl=http://aaa.ss.com/ss.xxx -DrepositoryId=release-repo
對于我們的工程輸出,如果需要放置到公司的repository中的話,可以通過配置pom來實現(xiàn)
Xml代碼
這里使用的scp方式提交庫文件,還有其他方式可以使用,請參考faq部分。然后記得在你的settings.xml中加入這一內容
Xml代碼
maven定義了很多變量屬性,參考這里 http://docs.codehaus.org/display/MAVENUSER/MavenPropertiesGuide
... ... hello
我們已經知道m(xù)aven預定義了許多的階段(phase),每個插件都依附于這些階段,并且在進入某個階段的時候,調用運行這些相關插件的功能。我們先來看完整的maven生命周期:
生命周期 階段描述validate | 驗證項目是否正確,以及所有為了完整構建必要的信息是否可用 |
generate-sources | 生成所有需要包含在編譯過程中的源代碼 |
process-sources | 處理源代碼,比如過濾一些值 |
generate-resources | 生成所有需要包含在打包過程中的資源文件 |
process-resources | 復制并處理資源文件至目標目錄,準備打包 |
compile | 編譯項目的源代碼 |
process-classes | 后處理編譯生成的文件,例如對Java類進行字節(jié)碼增強(bytecode enhancement) |
generate-test-sources | 生成所有包含在測試編譯過程中的測試源碼 |
process-test-sources | 處理測試源碼,比如過濾一些值 |
generate-test-resources | 生成測試需要的資源文件 |
process-test-resources | 復制并處理測試資源文件至測試目標目錄 |
test-compile | 編譯測試源碼至測試目標目錄 |
test | 使用合適的單元測試框架運行測試。這些測試應該不需要代碼被打包或發(fā)布 |
prepare-package | 在真正的打包之前,執(zhí)行一些準備打包必要的操作。這通常會產生一個包的展開的處理過的版本(將會在Maven 2.1+中實現(xiàn)) |
package | 將編譯好的代碼打包成可分發(fā)的格式,如JAR,WAR,或者EAR |
pre-integration-test | 執(zhí)行一些在集成測試運行之前需要的動作。如建立集成測試需要的環(huán)境 |
integration-test | 如果有必要的話,處理包并發(fā)布至集成測試可以運行的環(huán)境 |
post-integration-test | 執(zhí)行一些在集成測試運行之后需要的動作。如清理集成測試環(huán)境。 |
verify | 執(zhí)行所有檢查,驗證包是有效的,符合質量規(guī)范 |
install | 安裝包至本地倉庫,以備本地的其它項目作為依賴使用 |
deploy | 復制最終的包至遠程倉庫,共享給其它開發(fā)人員和項目(通常和一次正式的發(fā)布相關) |
maven核心的插件列表可以參考 http://maven.apache.org/plugins/index.html 。這里僅列舉幾個常用的插件及其配置參數(shù):
除了以下的幾個faq條目之外,還有一些faq可以參考
兄弟們如果有其他問題,歡迎跟帖提問!
答:直接在pom文件中加入一個dependency節(jié)點,如果要刪除依賴,把對應的dependency節(jié)點刪除即可。
答:設置exclusion即可。
Xml代碼
答:加入一個特殊的依賴關系,使用system類型,如下:
com.abc my-tools2.5.0 jar system ${basedir}/lib/mylib1.jar
但是要記住,發(fā)布的時候不會復制這個jar。需要手工配置,而且其他project依賴這個project的時候,會報告警告。如果沒有特殊要求,建議直接注冊發(fā)布到repository。
答:在project屬性中去掉java build path中對其他 project 的依賴關系,直接在pom中設置依賴關系即可
com.abc.project1 abc-project1-common${project.version}
另外,保證沒有其他錯誤。
答:使用 assembly 插件即可。
maven-assembly-plugin jar-with-dependencies
答:maven本身在發(fā)布的時候,可以發(fā)布單純的jar,也可以同時發(fā)布xxx-tests.jar和xxx-javadoc.jar(大家經常在repository中可以看到類似的東西)。我們自己的項目A要同時輸出test.jar可以做如下的設置
org.apache.maven.plugins maven-jar-plugintest-jar
然后在其他需要引用的工程B中做如下的dependency設置
Xml代碼
答:使用資源過濾功能,比如:
.. .. com.mysql.jdbc.Driver jdbc:mysql://localhost:3306/development_db dev_user s3cr3tw0rd .. src/main/resources true production oracle.jdbc.driver.OracleDriver jdbc:oracle:thin:@proddb01:1521:PROD prod_user s00p3rs3cr3t
答: maven-svn-revision-number-plugin 可以從 SVN 中獲取版本號,并將其變成環(huán)境變量,交由其他插件或者profile使用,詳細幫助在這里 。一般和resource的filter機制同時使用
com.google.code.maven-svn-revision-number-plugin maven-svn-revision-number-plugin1.3 revision prefix
這段代碼負責把resource文件中的內容替換成適當內容
repository = ${prefix.repository} path = ${prefix.path} revision = ${prefix.revision} mixedRevisions = ${prefix.mixedRevisions} committedRevision = ${prefix.committedRevision} status = ${prefix.status} specialStatus = ${prefix.specialStatus}
答:以下內容設定編譯器編譯java1.5的代碼
... ... ...... maven-compiler-plugin 1.5
要設置其他插件的參數(shù)也可以,請參考對應插件的幫助信息
答:指定source目錄和test-source目錄即可。
target src
這個例子把源代碼設置成了src目錄,測試代碼在test目錄,所以輸出到bin目錄。這里要注意,directory如果也設置成bin目錄的 話,maven打包的時候會引起死循環(huán),因為directory是所有工作存放的地方,默認包含outputDirectory定義的目錄在內。
http://www.iteye.com/topic/973166
聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com