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
感兴趣的小伙伴可关注个人公众号:壹枝花算不算浪漫

posted @ 2019-11-19 09:50  一枝花算不算浪漫  阅读(1680)  评论(0编辑  收藏  举报