• Maven使用教程一:Maven基础


    使用Maven快速创建一个工程

    为了加快速度,在setting.xml中加一段配置,用国内阿里云的镜像仓库可以去下载各种东西。

    <mirror>
        <id>nexus-aliyun</id>
        <mirrorOf>*</mirrorOf>
        <name>Nexus aliyun</name>      <url>http://maven.aliyun.com/nexus/content/groups/public</url>
    </mirror>
    
    

    然后测试下,在终端输入:

    进入代码保存的相关目录

    cd /Users/wangmeng/Documents/space/learn
    

    构建一个maven工程:

    mvn archetype:generate -DgroupId=com.wangmeng.maven -DartifactId=maven-first-app -DarchetypeArtifactId=maven-archetype-quickstart -DinteractiveMode=false
    

    此时maven会在当前目录下 新建一个目录,名称就是-DartifactId指定的名称。这个这个目录,就可以看到maven自动给我们初始化好了一个工程对应的目录结构。

    maven的约定

    这就是基本的maven工程目录结构,其中src/main/java目录包含了这个项目的java源码,src/test/java目录包含了测试代码,pom.xml文件就是maven的核心配置文件

    pom.xml初步介绍

    pom.xml文件是一个项目最核心的maven配置文件,包含了大量的信息,maven真是基于这里的配置信息来对工程进行构建管理工作的。一个最基本的pom.xml文件如maven生成的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/maven-v4_0_0.xsd">
      <modelVersion>4.0.0</modelVersion>
      <groupId>com.zhss.maven</groupId>
      <artifactId>maven-first-app</artifactId>
      <packaging>jar</packaging>
      <version>1.0-SNAPSHOT</version>
      <name>maven-first-app</name>
      <url>http://maven.apache.org</url>
      <dependencies>
        <dependency>
          <groupId>junit</groupId>
          <artifactId>junit</artifactId>
          <version>3.8.1</version>
          <scope>test</scope>
        </dependency>
      </dependencies>
    </project>
    
    
    • :pom.xml中的顶层元素
    • :POM本身的版本号,一般很少改变
    • :创建这个项目的公司或者组织,一般用公司网站后缀,比如com.company
    • :这个项目的唯一标识,一般生成的jar包名称,会是-.这个格式,比如说myapp-1.0.jar
    • :要用的打包类型,比如jar,war等等
    • :这个项目的版本号
    • :这个项目用于展示的名称,一般在生成文档的时候使用
    • :这是这个项目的文档能下载的站点url,一般用于生成文档
    • :用于项目的描述

    maven体系结构

    我们项目中的pom.xml是如何依赖以及构建的呢?

    全局的maven配置,settings.xml
    maven的约定:各种约定目录,代码/资源/输出/测试

    mvn构建命令:

    • maven一定会去考虑settings.xml配置文件里的一些配置
    • maven会去解析你的maven工程的pom.xml
    • maven会去看你的pom.xml里声明了哪些依赖
    • maven会去本地仓库里找有没有这些依赖,比如有没有junit
    • 如果本地仓库没有junit,那么就会去远程仓库去找,下载junit,所谓的运程仓库里包含了几乎所有的依赖包
    • 远程仓库下载到了junit以后,就会放到本地仓库,缓存起来,供你以后去使用,maven的远程中央仓库:https://repo.maven.apache.org/maven2/

    maven的体系结构图:
    maven体系结构图.png

    Maven坐标

    每个maven项目都有一个坐标
    groupId+artifactId+version+packaging+classifier 五个维度的坐标,唯一定义一个依赖包,任何一个项目,都是用这五个维度唯一定位一个发布包。

    • groupId:一般是公司或者组织的官网域名倒序来开头,比如com.baidu大头
    • artifactId:项目中的某个模块,或者某个服务名
    • version:工程的版本号
    • packaging:这个工程的发布包打包方式,一般常用的就是jar和war年终
    • classifier:很少用,定义某个工程的附属项目,比如hello-world工程的 hello-world-source工程,就是源码

    例如:
    groupId: com.wangmeng.oa
    artifactId: oa-organ
    version: 1.0.0-SNAPSHOT

    • version中第三位是小版本,一般修复了一些bug 或者轻微的改动会累加第三位小版本
    • 第二位是小版本,一般如果一路一些新的功能或模块,或者做了一些重构,会累加第二位小版本
    • 第一位是大版本,一般就是如果整体架构有特别的升级或者变化,才会累加第一位大版本。
    • SNAPSHOT,就是当前这个版本下的快照版本,代表代码正在开发或者测试中,可以试用,但是没有经过完善测试,不会承诺非常稳定

    Maven依赖管理机制

    <dependency>
            <groupId></groupId>
            <artifactId></artifactId>
            <version></version>
            <type></type>
            <scope></scope>
            <optional></optional>
    </dependency>
    

    1,依赖范围:

    • compile:默认,对编译、测试和运行的classpath都有效
    • test:仅仅对于运行的测试代码的classpath有效
    • provided:编辑和测试的时候有效,但是在运行的时候无效
    • runtime:测试和运行classpath有效,但是编译代码时无效

    2,传递性依赖

    maven的依赖性传递,就是说会自动递归解析所有的依赖,然后负责将依赖下载下来,所有层级的依赖都会成为我们项目的依赖。

    比如说我们依赖于A是compile范围,A依赖于B,B是test范围。那么A只有在测试的时候才会使用B。

    传递性依赖机制对依赖范围也是有影响的,比如下面的表格,第一列是一级依赖,第一行是二级依赖,传递性依赖会导致多级依赖的依赖范围交叉在一起,会有影响。

    3,依赖调解

    依赖传递性会导致依赖冲突的问题,例如

    比如A->B->C->X(1.0),A->D->X(2.0),A有两个传递性依赖X,不同的版本

    此时就会依赖调解,就近原则,离A最近的选用,就是X的2.0版本

    如果A->B->X(1.0)和A->D->X(2.0),路径等长呢?那么会选择第一声明原则,哪个依赖在pom.xml里先声明,就用哪个

    4,可选依赖
    <optional>true</optional> 此时以来传递失效,不会向上传递。

    Maven依赖冲突

    1,依赖冲突
    因为maven的依赖具有传递性,所以就应运而生了依赖的冲突
    比如我们同时依赖A和B,此时A依赖了C-1.0, B依赖了D,D依赖了C-2.0

    按照上面所说取最短路径,所以我们项目依赖C-1.0
    如果C-1.0 有方法CClass.sayHello()
    C-2.0中添加新方法CClass.printHello()

    D调用C-2.0中的printHello()这个方法,这时因为项目中依赖的是C-1.0,所以使用时就会报错

    2,解决依赖冲突
    mvn dependency:tree命令,查看项目中maven依赖关系树,然后将有冲突的依赖找出来,在pom中使用exclusion处理。

    A
      --C-1.0
    B
      --D
        --C-2.0

    <dependency>
    	<groupId>A</groupId>
    	<artifactId>A</artifactId>
    	<version>1.0</version>
    	<exclusions>
    	    <exclusion>
    		    <groupId>C</groupId>
    		    <artifactId>C</artifactId>
    	    </exclusion>
    	</exclusions>
    </dependency>
    

    Maven的多层仓库架构

    maven仓库的大类分为本地仓库和远程仓库两种,如果我们声明了一个依赖,那么在构建打包的时候,先会去本地仓库找,这个本地仓库的地址默认就是~/.m2/repository目录下面,当然settings.xml文件是可以修改这个地址的。如果本地仓库找不到,那么就会去远程仓库找,默认是去maven自己的中央仓库里找,maven的中央仓库几乎涵盖所有的依赖,然后会将中央仓库的依赖下载下来放到本地仓库,缓存起来,供下次使用。

    maven多层仓库架构.png
    感兴趣的小伙伴可关注个人公众号:壹枝花算不算浪漫

  • 相关阅读:
    python数据库小脚本
    替换debug
    监听属性变化
    ast binding
    还原未修改的常量
    数据库中常用查询
    职业生涯四个阶段
    如何控制需求
    如何将VS Code 切换成中文
    当同时使用bootstrapdatepicker.js和jquery.validate.js这两款插件,至少要选择两次时间,才能验证成功的问题
  • 原文地址:https://www.cnblogs.com/wang-meng/p/11887337.html
Copyright © 2020-2023  润新知