Spring Boot 多模块 Maven 项目的部署

3
我们有一个多模块的Maven项目,包括以下模块:
  • Commons(公共)
  • Model(模型)
  • Repository(仓库)
  • Service(服务)
  • Web(网页)
我们搜索了一下,没有找到在这种结构下如何制作一个可执行的jar包的解决方案。
这是父pom.xml文件:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>...</groupId>
    <artifactId>...</artifactId>
    <version>1.0-SNAPSHOT</version>
    <modules>
        <module>Model</module>
        <module>Web</module>
        <module>Service</module>
        <module>Repository</module>
        <module>Common</module>
    </modules>
    <packaging>pom</packaging>

    <name>Api</name>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <java.version>1.8</java.version>

        <mssql.jdbc.driver.version>4.2</mssql.jdbc.driver.version>

        <apache.commons.lang.version>3.3.2</apache.commons.lang.version>
        <apache.commons.collection.utils.version>4.1</apache.commons.collection.utils.version>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.4.1.RELEASE</version>
    </parent>

    <dependencies>

        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-envers</artifactId>
        </dependency>

        <dependency>
            <groupId>com.fasterxml.jackson.datatype</groupId>
            <artifactId>jackson-datatype-jsr310</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
        </dependency>

    </dependencies>

</project>

所以,这是一个非常基础的父pom.xml。我们知道当打包为pom时无法使用Spring Boot Maven插件,因此我们在Web模块的pom.xml中定义了它:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>...</artifactId>
        <groupId>...</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <groupId>...</groupId>
    <artifactId>...</artifactId>
    <packaging>jar</packaging>

    <name>...</name>

    <dependencies>

        <dependency>
            <groupId>...</groupId>
            <artifactId>Service</artifactId>
            <version>${project.version}</version>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <version>1.4.1.RELEASE</version>
                <configuration>
                    <mainClass>...App</mainClass>
                </configuration>
                <!--<executions>-->
                    <!--<execution>-->
                        <!--<goals>-->
                            <!--<goal>repackage</goal>-->
                        <!--</goals>-->
                    <!--</execution>-->
                <!--</executions>-->
            </plugin>
        </plugins>
    </build>

</project>

其他模块的pom.xml基本相同:Service的pom.xml包括Repository.jar,Repository包括Model和Common。

问题1:当我们运行mvn install时,插件会在web模块的目标文件夹中创建一个fat jar。但是,当我们尝试使用java -jar name-of-jar运行它时,它会给出有关来自Service模块的某个类的java.lang.NoClassDefFoundError错误 - 从Web的pom.xml可以看出Service模块已经被包括进去了(还有一个作为库打包在Web.jar中的Service.jar)。

问题2:有趣的是,在Web.jar中的Service.jar也包含了几乎所有已经存在于Web.jar中的依赖项 - 基本上是重复的。

有人能够使用Spring Boot Maven插件和Maven模块项目结构创建可执行的fat jar吗?


我有一种预感,开发工具可能是问题的根源(这也是我的答案)。NoClassDefFound意味着找到了类,但在初始化期间失败了。我猜测这可能是由于WAR和Service.jar中包含了重复的jar文件引起的。不同类加载器中可能存在类的不匹配,从而导致NoClassDefFound错误?如果即使移动devtools后错误仍然存在,请提供堆栈跟踪。 - alexbt
是的,这解决了两个问题,谢谢。 - Nikola Zarić
1个回答

1

问题1和2

您的父POM中不应将devtools作为依赖项。 请将以下内容移至web/spring-boot模块:

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-devtools</artifactId>
</dependency>

如果您查看 devtools 的 pom,它包括以下内容:
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-autoconfigure</artifactId>
</dependency>

因此,您的服务jar最终会带有一些Spring Boot依赖项。


我明白这就是重复依赖的原因,但将devtools依赖项移动到Web模块如何有助于构建可执行jar文件呢? - Nikola Zarić
你尝试过移动依赖项并重新构建吗? - alexbt

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接