在學(xué)習(xí)Spring框架的過(guò)程中肮塞,要導(dǎo)入很多jar包襟齿,不但非常麻煩,初學(xué)時(shí)還很容易忘記要用哪些jar包峦嗤,尤其是有依賴(lài)關(guān)系的jar包蕊唐。偶然看到有使用maven構(gòu)建web項(xiàng)目的方法屋摔,可以非常方便的管理jar包還可以烁设,就深入研究了一下。
一钓试、jar包的導(dǎo)入引發(fā)的問(wèn)題
在學(xué)習(xí)使用框架的時(shí)候装黑,要先導(dǎo)入一些jar包,比如:spring-webmvc-4.1.5.RELEASE.jar弓熏。
使用SSM框架時(shí)要導(dǎo)入十幾個(gè)jar包恋谭,需要手動(dòng)一個(gè)一個(gè)導(dǎo)入,非常繁瑣挽鞠,還容易遺忘有依賴(lài)關(guān)系的一些jar包疚颊,(比如,要使用A包信认,A依賴(lài)于B,C,D,要使用A,就要先導(dǎo)入B,C,D材义,有些依賴(lài)容易忘記,有些依賴(lài)可能根本不知道)嫁赏。
為了解決這種依賴(lài)關(guān)系其掂,可以使用maven來(lái)構(gòu)建和管理項(xiàng)目。
二潦蝇、maven是什么款熬?
maven是工具,是基于項(xiàng)目對(duì)象模型(POM project object model)攘乒,可以通過(guò)一小段描述信息(配置)來(lái)管理項(xiàng)目的構(gòu)建贤牛,報(bào)告和文檔的軟件項(xiàng)目管理工具。
我認(rèn)為maven的核心功能是合理描述項(xiàng)目間的依賴(lài)關(guān)系则酝,就是通過(guò)pom.xml文件的配置獲取jar包殉簸,而不用手動(dòng)去添加jar包,如果需要使用pom.xml來(lái)獲取jar包堤魁,那么首先該項(xiàng)目就必須為maven項(xiàng)目喂链,maven項(xiàng)就是在java項(xiàng)目和web項(xiàng)目的上面包了一層maven,本質(zhì)上java項(xiàng)目還是java項(xiàng)目妥泉,web項(xiàng)目還是web項(xiàng)目椭微,但是包了maven之后,就可以使用maven提供的一些功能了(比如:通過(guò)pom.xml添加jar包)盲链。
三蝇率、maven的安裝
(一)迟杂,手動(dòng)安裝maven
從官網(wǎng)下載,并解壓到一個(gè)目錄本慕。
-
配置環(huán)境變量排拷,打開(kāi)系統(tǒng)環(huán)境變量新建一個(gè)M2_HOME將解壓后的maven存放路徑放在此,然后在path路徑加入
M2_HOME\bin
路徑
在mac中加入~/.bash_profile中
-
輸入mvn-v檢測(cè)是否安裝成功
- 復(fù)制
M2_HOME/conf/setting.xml
文件到$user.home/.m2/setting.xml
锅尘,這樣用戶每次升級(jí)后就不要在次修改setting.xml文件了监氢。如果不設(shè)置setting.xml,就會(huì)按照默認(rèn)的設(shè)置執(zhí)行藤违。根目錄下的設(shè)置是全局的浪腐,用戶目錄下的設(shè)置只對(duì)當(dāng)前用戶起作用。
(二)顿乒,高版本的Eclipse,MyEclipse中已集成了maven议街,可以直接使用。
設(shè)置maven為自己安裝的版本
這兩種方式的區(qū)別是璧榄,自己安裝的可以自己設(shè)置配置文件特漩,并使用命令行來(lái)管理項(xiàng)目,而集成的配置是默認(rèn)好的骨杂,且使用不了命令行涂身。
四、倉(cāng)庫(kù)的概念
通過(guò)pom.xml中的配置腊脱,就能夠獲取到想要的jar包访得,但是這些jar是在哪里呢?我們從哪里獲取到的這些jar包陕凹?答案是倉(cāng)庫(kù)悍抑。
倉(cāng)庫(kù)分為:本地倉(cāng)庫(kù)、第三方倉(cāng)庫(kù)(私服)杜耙、中央倉(cāng)庫(kù)
- 本地倉(cāng)庫(kù)搜骡,Maven會(huì)將工程中依賴(lài)的構(gòu)件(Jar包)從遠(yuǎn)程下載到本機(jī)一個(gè)目錄下管理,每個(gè)電腦默認(rèn)的倉(cāng)庫(kù)是在 $user.home/.m2/repository下
- 第三方倉(cāng)庫(kù)佑女,又稱(chēng)為內(nèi)部中心倉(cāng)庫(kù)记靡,也稱(chēng)為私服。一般是由公司自己設(shè)立的团驱,只為本公司內(nèi)部共享使用摸吠。它既可以作為公司內(nèi)部構(gòu)件協(xié)作和存檔,也可作為公用類(lèi)庫(kù)鏡像緩存嚎花,減少在外部訪問(wèn)和下載的頻率寸痢。(使用私服為了減少對(duì)中央倉(cāng)庫(kù)的訪問(wèn))。注意:連接私服紊选,需要單獨(dú)配置啼止。如果沒(méi)有配置私服道逗,默認(rèn)不使用
- 中央倉(cāng)庫(kù),Maven內(nèi)置了遠(yuǎn)程公用倉(cāng)庫(kù):
http://repo1.maven.org/maven2
献烦。這個(gè)公共倉(cāng)庫(kù)是由Maven自己維護(hù)滓窍,里面有大量的常用類(lèi)庫(kù),并包含了世界上大部分流行的開(kāi)源項(xiàng)目構(gòu)件巩那。目前是以java為主吏夯,工程依賴(lài)的jar包如果本地倉(cāng)庫(kù)沒(méi)有,默認(rèn)從中央倉(cāng)庫(kù)下載拢操。
五锦亦、在MyEclipse中使用maven創(chuàng)建web項(xiàng)目
步驟
-
右鍵,new maven project
-
選中Create a simple project
-
填寫(xiě)項(xiàng)目名
或者選擇用模板創(chuàng)建項(xiàng)目
-
不選Create a simple project令境,直接下一步,在彈出的對(duì)話框中選擇類(lèi)型
-
填寫(xiě)項(xiàng)目名
創(chuàng)建完項(xiàng)目后顾瞪,有幾點(diǎn)要注意:
-
更改jre版本
-
如果是使用simple project創(chuàng)建項(xiàng)目舔庶,則需要生成web.xml文件(使用模板創(chuàng)建的項(xiàng)目會(huì)自動(dòng)生成)
項(xiàng)目結(jié)構(gòu)如下
此時(shí)正常部署項(xiàng)目到服務(wù)器,并運(yùn)行:
可能出現(xiàn)的問(wèn)題
1)創(chuàng)建maven project時(shí)提示內(nèi)存不足陈醒,要修改MyEclipse安裝目錄下的eclispe.ini 文件惕橙,把內(nèi)存分配修改一下(如:1024m改成2048m)
2)pom.xml文件報(bào)錯(cuò): Cannot detect Web Project version,
這是因?yàn)閟imple project 要指定編譯war的版本钉跷,在pom.xml中加入以下代碼:
在這里指定JRE版本是因?yàn)槊逐校褂胢aven update項(xiàng)目時(shí),會(huì)在配置文件中查找版本號(hào)爷辙,默認(rèn)的是1.5彬坏,如果你用的是1.7或1.8,沒(méi)有在這里寫(xiě)明膝晾,就會(huì)update 項(xiàng)目后變成1.5而導(dǎo)致其他問(wèn)題栓始。
3)jsp報(bào)錯(cuò),The superclass "javax,servlet.http.HttpServlet" was not found on the Java Build Path
在pom.xml添加依賴(lài):
4)導(dǎo)入maven項(xiàng)目時(shí)血当,Referenced file contains errors幻赚,這個(gè)錯(cuò)一般發(fā)生在導(dǎo)入maven項(xiàng)目時(shí),
這時(shí)臊旭,只需update project即可:
5)Cannot upgrade/downgrade to Dynamic Web Module 3.0 facet.
解決這個(gè)問(wèn)題要更改一下web.xml中的版本約束:
6)WEB-INF文件夾下lib中沒(méi)有jar包的問(wèn)題:
maven用dependency來(lái)引用倉(cāng)庫(kù)中的jar包落恼,所以,在lib文件夾下是沒(méi)有jar包的离熏,如果佳谦,我想引用自己寫(xiě)的一個(gè)jar包,就可以放在lib下撤奸,在部署到服務(wù)器時(shí)吠昭,會(huì)一同部署過(guò)去:
六喊括、pom.xml的依賴(lài)關(guān)系
在pom.xml中添加dependency即可引入jar包
比如:我想添加spring-webmvc的jar包到項(xiàng)目中,可以在mavenRepository中央倉(cāng)庫(kù)中查找到相應(yīng)的版本矢棚,把代碼直接拷貝到pom.xml中即可:
dependency有9個(gè)屬性郑什,或子節(jié)點(diǎn)
groupId ---> 組名,一般是反向域名蒲肋,基本屬性蘑拯,不可少
artifactId ---> 項(xiàng)目名,基本屬性兜粘,不可少
version ---> 版本申窘,基本屬性,不可少
type ---> 依賴(lài)的類(lèi)型孔轴,比如是jar包還是war包等剃法,默認(rèn)為jar
classifier ---> 定義構(gòu)件輸出的一些附屬構(gòu)件
-
scope ---> 依賴(lài)范圍,意思就是通過(guò)pom.xml加載進(jìn)來(lái)的jar包路鹰,來(lái)什么范圍內(nèi)使用生效贷洲,范圍包括編譯時(shí),運(yùn)行時(shí)晋柱,測(cè)試時(shí)优构。
- compile:默認(rèn)值,如果選擇此值雁竞,表示編譯钦椭、測(cè)試和運(yùn)行都使用當(dāng)前jar
- test:表示只在測(cè)試時(shí)當(dāng)前jar生效,在別的范圍內(nèi)就不能使用該jar包碑诉。
- runtime彪腔,表示測(cè)試和運(yùn)行時(shí)使用當(dāng)前jar,編譯時(shí)不用該jar包联贩。例如:JDBC驅(qū)動(dòng)漫仆。JDBC驅(qū)動(dòng),在編譯時(shí)(也就是我們寫(xiě)代碼的時(shí)候都是采用接口編程泪幌,壓根就沒(méi)使用到JDBC驅(qū)動(dòng)包內(nèi)任何東西盲厌,只有在運(yùn)行時(shí)才用的到,所以這個(gè)是典型的使用runtime這個(gè)值的例子)祸泪,此處不寫(xiě)也不報(bào)錯(cuò)吗浩,理由同上
- provided,表示編譯和測(cè)試時(shí)使用當(dāng)前jar没隘,運(yùn)行時(shí)不在使用該jar了懂扼。例如:servlet-api、jsp-api等》【必須填寫(xiě)】什么意思呢赶熟? 在我們以前創(chuàng)建web工程,編寫(xiě)servlet或者jsp時(shí)陷嘴,沒(méi)導(dǎo)入過(guò)jar包映砖,因?yàn)閙yeclipse幫我們提供了這兩個(gè)jar包,內(nèi)置了灾挨,所以我們?cè)诰幾g期測(cè)試期使用servlet都不會(huì)報(bào)缺少jar包的錯(cuò)誤邑退,而在運(yùn)行時(shí)期,離開(kāi)了myeclipse劳澄,就相當(dāng)于缺失了這兩個(gè)jar包地技,但此時(shí)tomcat又會(huì)幫我們提供這兩個(gè)jar,以便我們不會(huì)報(bào)錯(cuò)秒拔。如果使用maven開(kāi)發(fā)項(xiàng)目莫矗,就不是web項(xiàng)目了,那么myeclipse就不會(huì)在給我們提供這兩個(gè)jar包溯警,我們就必須自己手動(dòng)通過(guò)坐標(biāo)從倉(cāng)庫(kù)中獲取趣苏,但是針對(duì)上面的分析,當(dāng)運(yùn)行的時(shí)候梯轻,tomcat會(huì)幫我們提供這兩個(gè)jar包,所以我們自己從倉(cāng)庫(kù)中獲取的jar包就不能和tomcat中的沖突尽棕,那么就正好可以通過(guò)provided這個(gè)屬性喳挑,來(lái)設(shè)置這兩個(gè)jar的作用范圍,就是在變異時(shí)期和測(cè)試時(shí)期生效即可滔悉。
- system:表示我們自己手動(dòng)加入的jar包伊诵,不屬于maven倉(cāng)庫(kù)(本地,第三方等)回官,屬于別得類(lèi)庫(kù)的這樣的jar包曹宴,只在編譯和測(cè)試期生效,運(yùn)行時(shí)無(wú)效歉提。一般不用
systemPath ---> 當(dāng)maven依賴(lài)本地而非repository中的jar包笛坦,sytemPath指明本地jar包路徑
exclusions ---> 排除傳遞依賴(lài),解決jar沖突問(wèn)題苔巨。
依賴(lài)傳遞的意思就是版扩,A項(xiàng)目 依賴(lài) B項(xiàng)目,B項(xiàng)目 依賴(lài) C項(xiàng)目侄泽,當(dāng)使用A項(xiàng)目時(shí)礁芦,就會(huì)把B也給加載進(jìn)來(lái),這是傳遞依賴(lài),依次類(lèi)推柿扣,C也會(huì)因此給加載進(jìn)來(lái)肖方。這個(gè)有依賴(lài)傳遞有好處,也有壞處未状,壞處就是jar包的沖突問(wèn)題俯画,比如,A 依賴(lài) B(B的版本為1)娩践,C 依賴(lài) B(B的版本為2)活翩,如果一個(gè)項(xiàng)目同時(shí)需要A和C,那么A,C都會(huì)傳遞依賴(lài)將B給加載進(jìn)來(lái)翻伺,問(wèn)題就在這里材泄,兩個(gè)B的版本不一樣,將兩個(gè)都加載進(jìn)去就會(huì)引起沖突吨岭,這時(shí)候就需要使用exclusions這個(gè)屬性配置了拉宗。optional ---> 標(biāo)記依賴(lài)是否可選。默認(rèn)值false
依賴(lài)調(diào)節(jié)的兩個(gè)原則
這個(gè)就是maven解決傳遞依賴(lài)時(shí)jar包沖突問(wèn)題的方法:
1辣辫、第一原則:路徑近者優(yōu)先原則
A-->B-->C-->D-->X(1.6)
E-->D-->X(2.0)
使用X(2.0)旦事,因?yàn)槠渎窂礁?/p>
2、第二原則:第一聲明者優(yōu)先原則急灭。就是如果路徑相同姐浮,maven 默認(rèn)配置在前面的優(yōu)先使用
A-->B--> X(1.6)
C-->D--> X(2.0)
這樣就是路徑相同,那么如果A在前面葬馋,C在后面卖鲤,則使用X(1.6)
maven會(huì)先根據(jù)第一原則進(jìn)行選擇,第一原則不成畴嘶,則按第二原則處理蛋逾。