前言
Maven 的哲學(xué)思想是懊缺,約定優(yōu)于配置(Convention Over Configuration)墅冷,Maven 依賴中 scope 的默認(rèn)值是compile
Scope 指定了依賴(第三方j(luò)ar包)的 作用范圍
作用范圍包括派撕,所在項目的測試、編譯、運行、打包等生命周期
其中尊沸,編譯和運行還分為
測試代碼的編譯和運行
非測試代碼的編譯和運行
scope 分類
test 測試范圍
測試范圍的依賴(第三方j(luò)ar包)威沫,針對測試相關(guān)代碼的編譯和運行,在通常代碼的編譯和運行時都不需要洼专,只有在有關(guān)測試的代碼編譯和運行測試代碼階段可用
案例說明
環(huán)境 IJ maven web project
前提:需知facade slf4j 只有在classpath 中發(fā)現(xiàn)底層日志實現(xiàn)framework時才起作用
pom.xml
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<!--此處test,表明 junit jar包只能出現(xiàn)在 test 環(huán)境下的classpath,
即 只能在標(biāo)記為 “TestSources Root的 文件夾下”被調(diào)用, 而在“標(biāo)記為Sources Root 的文件夾下”找不到其中的類-->
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.7.25</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-simple</artifactId>
<version>1.7.25</version>
<!--此處test 是關(guān)鍵棒掠,表明底層日志實現(xiàn)框架-也即該依賴在test 代碼編譯、運行時才起作用-->
<scope>test</scope>
</dependency>
</dependencies>
測試代碼
測試方法之一壶熏,即是上面pom注釋句柠,junit 類調(diào)用只能在TestSources Root 目錄下起作用
測試方法之二
Sources Root 目錄下
public class LogDemo {
Logger logger = LoggerFactory.getLogger(LogDemoTest.class);
public static void main(String[] args) {
Logger logger = LoggerFactory.getLogger(LogDemo.class);
//不能找到底層日志使系統(tǒng),因為 該依賴 為 test 范圍
logger.info("log");
}
}
輸出:
SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
TestSources Root 目錄下
public class LogDemoTest {
Logger logger = LoggerFactory.getLogger(LogDemoTest.class);
@Test
public void test() {
//能找到底層日志實現(xiàn)系統(tǒng)
logger.info("junit test");
}
}
輸出:
[main] INFO log.learn.LogDemoTest - junit test
compile 編譯范圍
依賴默認(rèn)范圍棒假,該依賴需要參與當(dāng)前項目的編譯、測試精盅、運行帽哑、打包
什么叫參與當(dāng)前項目的編譯?叹俏?妻枕?
runtime 運行時范圍
表示依賴無需參與當(dāng)前項目的編譯,但是后期的運行和測試需要參與粘驰,不參與編譯能運行成功嗎屡谐??
與 compile 相比蝌数,跳過編譯而已
比如愕掏,你可能在編譯的時候需要 JDBC API JAR,只有在 運行時才需要 JDBC
貌似是編譯時該包不參與顶伞,運行時參與饵撑?
provided 表明該依賴已經(jīng)提供,故只在未提供時才被使用
應(yīng)用場景是你定義了一個Servlet唆貌,此刻得需要Servlet-api.jar 才能編譯成功滑潘,但是當(dāng)你達(dá)成war 包時,你并不想將 Servlet-api.jar 包進(jìn)去锨咙,因為Tomcat等容器會提供
跟compile 類似语卤,說明JDK、容器或使用者會提供這個依賴酪刀,如Servlet.jar
這個依賴只作用在** 編譯和測試粹舵,該依賴會由系統(tǒng)組件提供,不需手動添加蓖宦,只存在編譯齐婴、運行、測試階段稠茂,打包是不用包進(jìn)去柠偶,打包階段做了exclude**動作
沒有傳遞性
system
被依賴項不會從maven倉庫下載情妖,而是從本地系統(tǒng)指定路徑下尋找,需要 systemPath 屬性
scope 的傳遞依賴
A -> B -> C, 當(dāng)前項目 A诱担,A依賴于B毡证,B依賴于C,知道B在 A中的scope蔫仙,怎么知道 C在 A 中的 scope
即料睛,A需不需要 C的問題,本質(zhì)由 C在B中的scope決定
當(dāng) C 在 B 中的scope 是test 或 provided 時摇邦,C 直接被丟棄恤煞,A不依賴C
否則 A 依賴 C,C的scope 繼承與B 的scope