title: Maven 依赖 install
date: 2018-11-08 16:23:37
tags:
基础
首先明确:
- mvn install 出来的 jar 包只会包含自己的 src 的 classes. 即使你是 compile 的依赖, 也不会进去, 但是如果打成 war 包, 是会包含 compile scope 的依赖的. 而 provided 是要容器提供, 比如说 Tomcat, 会到 Tomcat 的
$liferay-tomcat-home\webapps\ROOT\WEB-INF\lib
目录下找. - 可以通过 assembly/shade 插件把依赖的 jar 包打到一个 assembly.jar 包中去. 和源码的 jar 包可以是独立的, 也可以打到一起. 如果你一个依赖(D1)有两个版本(在父/子pom 中都有定义, 但是版本不一样), 在打出的 jar 包里只会有一个版本, 因为路径里不带版本的. 所以会出现各种 NoSuchMethodError 等等问题, 因为编译的时候都是各自用的正确的 D1 编译的出的 class. 但是运行时用到的 D1 只会有一个版本, 会有不匹配.
所以, 不要在一个项目里, 不同 pom 里面尝试使用不同 version 的依赖. 来看个实例:
parquet-column 里会 shade 一个 fasttuil, 你 jar -tf parquet-column.jar 看他 里面会有这个 fastutil.
可以看到有两个 jar 包, 一个 origin 不带 shade 的 fastutil, 另外一个是带着的, 也是放到 maven 仓库的 jar 包.
jar -tf 确认
问题定位
- 如果有遇到什么 NoSuchMethodError, ClassNotFoundException 等等的, 先看看打印出来的 classpath. IDEA 里可以直接看:
- 然后可以 double shift, 搜下出问题的类, 一般会跳出来多个:
- 然后再用
mvn dependency:tree
看下当前 model 用的哪个版本的依赖
然后就可以做相应的操作, 一般有以下几种:
1. exclusive 相应依赖
2. 写死用一个版本的
3. 把 dependency 的依赖做 external 模块, 然后 shade + reloaction, 可以参见 spark 的 external model.
番外
我们经常回用到 -pl :moduleName
, 看着很奇怪, 其实:前面省略的是 groupId.