Bootstrap

Spring Boot(一)— Maven

 Maven是一个跨平台的项目管理工具,主要服务于基于Java平台的项目构建、依赖管理和项目信息管理。通过其项目对象模型(POM),可以自动下载和管理项目所需的库和插件。

1 Maven

以上的pom.xml 包含了在开发中常用的标签与功能。

<?xml version="1.0" encoding="UTF-8"?>
<!---->
<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">
  <!--POM模型版本-->
  <modelVersion>4.0.0</modelVersion>
  
  <!--被继承的父项目,注意,如果没有显示标注父项目,那么其默认会继承Maven内置的一个POM(称为Super POM)-->
  <parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>3.2.4</version>
   </parent>
  
  <!--项目信息:组织ID、构建ID、版本号、项目名称-->    
  <groupId>com.huangmingfu</groupId>
  <artifactId>mvn_4</artifactId>	
  <version>0.1-SNAPSHOT</version>
  <name>mvn_4</name>
  
  <!--项目的打包方式,决定了项目构建的输出类型。常见的有:jar、war、pom、rar、maven-plugin、bundle等-->
  <packaging>jar</packaging>
  
  <!--定义属性,可以在pom中使用,也可以在pom包含的资源文件中使用-->
  <properties>
     <pom.directory>project-resources</pom.directory>
     <pom.info>pom中的描述</pom.info>
  </properties>
  
  <!--构建项目需要的信息-->
  <build>
    <!--被编译过的应用输出目录(不是生成的jar文件所在的目录,该目录由对应的插件配置来指定)-->
    <outputDirectory>./customTarget</outputDirectory>
    <!--构建产生的所有文件存放的目录 -->
    <directory>./customTarget</directory>
    <!--产生的构件的文件名,默认值是${artifactId}-${version}。 -->
    <finalName>生成文件</finalName>
    
     <!--定义构建生命周期中的扩展点。可以定义多个extension-->
    <extensions></extensions>
        
    <!--项目相关的所有资源路径,例如属性文件。这些资源被包含在最终的打包文件里-->
    <resources>
      <!--项目相关的资源路径-->
      <resource>
        <!--资源存放路径,相对于pom路径-->
        <directory>${pom.directory}</directory>
        <!--资源的目标(输出)路径,相对于outputDirectory指定的目录-->
        <targetPath>./</targetPath>
        <!--是否使用参数值来代替参数名。 参数值取自properties元素或文件里配置的属性,文件在filters元素里列出-->
        <filtering>true</filtering>
        <!--包含的文件列表模式-->
        <includes>
          <include>*.txt</include>
          <include>*.properties</include>
        </includes>
        <!--不包含的文件列表模式-->
        <excludes>
          <exclude>val*.properties</exclude>
        </excludes>
      </resource>
    </resources>
    
    <!--测试相关的所有资源路径-->
    <testResources></testResources>
    
    <!--当resource中的<filtering> 为true时生效。属性文件列表-->
    <filters>
      <!--路径相对于pom-->
      <filter>${pom.directory}/val.properties</filter>
      <filter>${pom.directory}/val2.properties</filter>
    </filters>
    
    <!--插件-->
    <plugins>
      <!--配置读取文件的编码-->
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>  
        <artifactId>maven-resources-plugin</artifactId>  
        <version>3.2.0</version>  
        <configuration>  
            <encoding>UTF-8</encoding>  
        </configuration> 
      </plugin>
    </plugins>
    
    <!--与dependencyManagement元素类似,用于管理插件。 可以在其中声明插件及配置,但不会出发调用,只有当配置了plugin元素,-->
    <!--并且其groupId和artifactId与其匹配时,才会影响实际的插件行为。常用于父项目确保所有子项目使用的插件版本保持一致-->
    <pluginManagement></pluginManagement>
  </build>  
  
  <!--配置文件,可以在构建中被激活,以改变项目的默认构建行为。通常用于处理跨环境差异(开发环境、生成环境等),在命令行中使用参数-P激活:mvn package -Pdev-->
  <profiles>
      <profile>
          <!--该配置文件的唯一标识-->
          <id>dev</id>
          <!--定义什么情况下应该激活profile,可以基于系统属性、环境变量、文件存在性、操作系统、JDK版本等条件-->
          <activation></activation>        
          <!--还可以包含properties、dependencies、build、modules、reporting等元素--> 
      </profile>
  </profiles>
  
  <!--子模块,被构成项目的一部分。模块元素值是指向该模块的相对路径-->
  <modules></modules>
  
  <!--项目相关的依赖-->
  <dependencies></dependencies>
  
  <!--依赖管理,常用于父项目确保所有的子项目使用的依赖版本保持一致-->
  <dependencyManagement></dependencyManagement>
</project>

1.1 基础信息

<version> 元素来指定项目的版本号。通常版本信息遵循:主版本.次版本.增量版本-里程碑版本。

主版本:项目的重大架构变更。

次版本:较大范围的功能增加或变化,以及bug修复。

增量版本:中大bug的修复。

里程碑版本:一个版本的里程碑,可能并不是一个非常稳定的版本,还需要很多测试。

Final

正式稳定版。

Beta

测试版。已经在开发团队内部进行了初步测试,但仍可能存在一些小问题,会提供给外部测试人员或团队进行更广泛的测试。

Alpha

内部测试版。通常处于开发阶段。用于内部进行初步功能测试和开发。

CR

候选发布版,会在公开发布之前进行最后的广泛测试。

SNAPSHOT

表示项目在开发中或不稳定的版本。 适用于开发和测试阶段。

表 常见的里程碑版本后缀

1.1.1 生命周期

Maven的生命周期是抽象的,本身不做任何实际的工作,实际工作由插件完成。

clean

清理项目,主要用于删除之前生成的数据。包含pre-clean、clean和post-clean三个阶段。

default

Maven的主要生命周期。包含了编译、测试、打包等一系列构建步骤。主要阶段有validate(验证工程是否正确、所需的资源是否可用)、compile(编译项目的源代码)、test(测试编译后的代码)、package(把已编译的代码打包成可发布的格式)、verify(运行所有检查)、install(将包安装至本地仓库,以供本地其他项目使用)

site

建立和发布项目站点。主要阶段有pre-site、site、post-site、site-deploy

表 Maven 的三套相互独立的生命周期

1.2 依赖

Maven中,依赖是指项目所依赖的库或框架,通常以jar包的形式存在。通过Maven坐标(groupId、artifactId、version)进行唯一标识。

Mavan支持多种依赖范围,用于控制依赖在项目不同阶段的可用性。

compile

默认范围,表示该依赖在编译、测试和运行阶段都可用。

test

仅在测试阶段可用。

runtime

在运行时可用,但在编译时不可用。

provided

该依赖在JDK或容器中已提供,无需在项目中打包。

system

与provided类似,但依赖的jar包必须显示地提供系统路径。

表 Maven依赖的范围

1.2.1 依赖的传递性

当项目A依赖与项目B,而项目B依赖项目C,那么A将自动继承项目B对项目C的依赖。

但是可能会导致一些问题,比如依赖冲突:

当项目中存在多个版本的相同依赖时,会根据一定的规则进行冲突解决。通常遵循“第一声明者优先”,即先声明的依赖版本将被使用。还可以使用<excusions>来排除特定的依赖,或者使用<dependencyManagement>来锁定依赖版本。

1.3 插件

插件是Maven的核心组成部分,实际上是一组用于执行特定任务的工具。可以自动化项目的构建、测试、打包、部署等过程。本质上是一个包含实现特定功能代码的jar包。

Maven的三大生命周期:clean、default和report,每个都包含了一系列的阶段(phase),而插件的目标(goal)则绑定在这些阶段上。当执行某个Maven命令时,实际上触发了绑定的插件目标。

maven-compiler-plugin

编译Java项目的源代码,并能确保Java代码与所指定的Java版本兼容,保证顺利地编译和执行。

maven-resources-plugin

处理项目中的资源文件,如复制资源文件到输出目录。可以在构建过程中复制和过滤资源文件。

maven-jar-plugin

将项目打包成JAR文件。

maven-war-plugin

将项目打包成WAR文件。

maven-install-plugin

将项目构建的包安装到本地Maven仓库中。

maven-dependency-plugin

用于分析和管理项目的依赖。可以执行复制依赖项到指定目录、列出项目依赖等任务。

表 Maven官方常见插件

1.4 打包

将项目打包成JAR或WAR文件时,会生成MANIFEST.MF文件(位于JAR文件等META-INFO目录下),用于描述打包等元数据信息(如程序入口类)。Java程序运行时,JVM读取并根据其中的配置信息执行响应的操作。

可以在maven-jar-plugin插件来配置其内容。

<packaging>标签指定打包方式,决定了文件输出类型(JAR、WAR等)。

1.4.1 <layout>

常用于插件的配置,用于定义打包过程的文件布局,不同的布局可能会影响如何组织JAR或WAR文件内的类、资源和依赖。

JAR

包含所有必要的依赖和启动脚本,生成的JAR文件将是可执行的。

WAR

与JAR类型类似,但适用Web应用。

ZIP

创建一个标准的ZIP文件,其内容和布局由用户自定义,不保证生成的ZIP文件是可执行的。

NONE

用于创建一个不包含任何启动脚本或布局的JAR或WAR文件。包含应用的类、资源及依赖。生成的文件不可执行。

CUSTOM

允许用户自定义布局。

表 <layout>的布局

2 spring-boot-starter-parent

用于构建Spring Boot应用程序的父项目,提供了一种标准化的方式来管理和配置Spring Boot应用程序的依赖项、插件和默认设置。

提供了以下功能(3.2.4版本):

1)默认使用Java17来编译项目。

2)UTF编码。

3)使用- parameters编译。

4)依赖管理。

5)repackage目标的执行,以便创建一个可执行的JAR或WAR文件。

6)Native配置文件。

7)filter资源处理。

8)合理的插件配置。包括Git提交ID和Shade插件的合理配置。

9)对应application.properties和application.yml的资源filter。

2.1 spring-boot-dependencies

不想使用spring-boot-starter-parent,但想使用其依赖管理来控制依赖版本时,可以使用spring-boot-dependencies依赖(是spring-boot-starter-parent的父级依赖)。可以设置项目的父pom为这个依赖,或者在依赖管理中引入这个依赖。

<dependencyManagement>

    <dependencies>

        <dependency>

            <!-- Import dependency management from Spring Boot -->

            <groupId>org.springframework.boot</groupId>

            <artifactId>spring-boot-dependencies</artifactId>

            <version>3.2.4</version>

            <type>pom</type>

            <scope>import</scope>

        </dependency>

    </dependencies>

</dependencyManagement>

2.2 依赖版本覆盖

当想在项目中覆盖某依赖在父POM中指定的版本时,可以在<properties>中覆盖该属性的值。

<properties>
    <mysql.version>8.1.0</mysql.version>
</properties>

或者在<dependencyManagement>中覆盖该依赖。

<dependencyManagement>
    <dependencies>
        <dependency>
            <groupId>com.mysql</groupId>
            <artifactId>mysql-connector-j</artifactId>
            <version>8.0.32</version>
            <exclusions>
                <exclusion>
                    <groupId>com.google.protobuf</groupId>
                    <artifactId>protobuf-java</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
    </dependencies>
</dependencyManagement>
;