Scope
compile(默认)
含义:compile 是默认值,如果没有指定 scope 值,该元素的默认值为 compile。被依赖项目需要参与到当前项目的编译,测试,打包,运行等阶段。打包的时候通常会包含被依赖项目。
provided
含义:被依赖项目理论上可以参与编译、测试、运行等阶段,相当于compile,但是在打包阶段做了exclude的动作。
适用场景:例如, 如果我们在开发一个web 应用,在编译时我们需要依赖 servlet-api.jar,但是在运行时我们不需要该jar包,因为这个jar包已由应用服务器提供,此时我们需要使用provided进行范围修饰。
runtime
含义:表示被依赖项目无需参与项目的编译,但是会参与到项目的测试和运行。与compile相比,被依赖项目无需参与项目的编译。
适用场景:例如,在编译的时候我们不需要 JDBC API 的 jar 包,而在运行的时候我们才需要 JDBC 驱动包。
maven的scope决定依赖的包是否加入本工程的classpath下:
依赖范围(Scope) | 编译classpath | 测试classpath | 运行时classpath | 传递性 | 说明 |
compile | Y | Y | Y | Y | |
test | - | Y | - | - | |
provided | Y | Y | - | - | |
runtime | - | Y | Y | Y | |
system | Y | Y | - | Y |
Optional
Maven optional关键字透彻图解 - 日拱一兵 - 博客园
两者区别
与scope区别在于:仅限制依赖包的传递性,不影响依赖包的classpath,看两个例子
比如一个工程中
A->B, B->C(scope:compile, optional:true),B的编译/运行/测试classpath都有C,A中的编译/运行/测试classpath都不存在C(尽管C的scope声明为compile),A调用B哪些依赖C的方法就会出错。
A->B, B->C(scope:provided), B的编译/测试classpath有C,A中的编译/运行/测试classpath都不存在C,但是A使用B(需要依赖C)的接口时就会出现找不到C的错误,此时,要么是手动加入C的依赖,即A->C,否则需要容器提供C的依赖包到运行时classpath。