在多模块 Maven 项目中共享公共模块

Share common module in multi-module Maven project

我有一个包含三个模块的项目:A、B 和 COMMON。 我想将通用逻辑放在 COMMON 模块(例如模型)中,然后是其他两个依赖于它的模块。

两个模块(A,B)将分别构建以创建两个不同的 jar。

我正在测试它试图将日志依赖项放入 COMMON 模块中,然后构建 A 项目。它会正确构建,但如果我尝试 运行 带有 "java -jar a.jar" 的 jar,它会失败并显示 NoClassDefFound:

Exception in thread "main" java.lang.NoClassDefFoundError: org/slf4j/LoggerFactory

根 pom:

<modules>
    <module>common</module>
    <module>a</module>
</modules>

<dependencyManagement>
    <dependencies>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-api</artifactId>
            <version>1.7.12</version>
        </dependency>
        <dependency>
            <groupId>ch.qos.logback</groupId>
            <artifactId>logback-classic</artifactId>
            <version>1.1.3</version>
        </dependency>
    </dependencies>
</dependencyManagement>

常用pom:

<dependencies>
    <dependency>
        <groupId>org.slf4j</groupId>
        <artifactId>slf4j-api</artifactId>
    </dependency>
    <dependency>
        <groupId>ch.qos.logback</groupId>
        <artifactId>logback-classic</artifactId>
    </dependency>
</dependencies>

和 pom:

<build>
    <plugins>
        <plugin>
            <groupId>org.apache.maven.plugins</groupId>
            <artifactId>maven-jar-plugin</artifactId>
            <version>2.4</version>
            <configuration>
                <archive>
                    <manifest>
                        <addClasspath>true</addClasspath>
                        <classpathPrefix>lib/</classpathPrefix>
                        <mainClass>bla.bla.Main</mainClass>
                    </manifest>
                </archive>
            </configuration>
        </plugin>
    </plugins>
</build>

<dependencies>
    <dependency>
        <groupId>bla.blu</groupId>
        <artifactId>common</artifactId>
        <version>0.0.1-SNAPSHOT</version>
    </dependency>
</dependencies>

在尝试 运行 来自 A.jar 依赖于通用功能的东西时,您需要在类路径中包含 COMMON 模块。简单地将其声明为依赖项不会在 a.jar 工件中包含常见的 类(如果需要,您需要使用程序集 Maven 插件)。

此外,common jar带来的所有传递依赖也应该在classpath中列出。

所以你必须 运行 就像 java -cp common.jar:slf4j-a.b.c.jar:logback-x.y.jar[:...] -jar a.jar

Maven 本身和任何知道 Maven 的 IDE 都会为您做这件事,所以您不必手动做。要在生产环境中构建 运行 的单个 jar,您应该考虑使用 assembly or the shade 插件并构建一个 "fat"(或 uber-)jar,它将在单个文件中包含所有依赖项。

默认情况下,Maven 不会将所有依赖项打包到您的 JAR 中。因此它不见了。当你 运行 它时,你应该在类路径上提供那个 jar。

其他选项是创建所谓的 fat jar。 This is SO thread how to do that

顺便说一句,我支持@Boris the Spider 关于 Spring Boot 的评论。走那条路,很多问题都会开箱即用。

编辑 - 对第一条评论的反应:

通常,公共依赖项应部署到工件库(Nexus、Artifactory)中,并作为任何第三方依赖项从那里获取。

因为你没有这样做,你需要在构建 A 期间在本地存储库中具有共同的依赖性。所以将 Spring-boot-maven-plugin 放入 A 构建并使用你的 multi- 构建 fat jar模块项目。