[size=medium][color=red]20. 项目版本[/color][/size]

[color=blue] 一个Maven项目发布版本号用version编码,用来分组和排序发布。Maven中的版本包含了以下部分:主版本,次版本,增量版本,和限定版本号。一个版本中,这些部分对应如下的格式:[/color]

<major version>.<minor version>.<incremental version>-<qualifier>

[color=blue] 例如:版本“1.3.5”由一个主版本1,一个次版本3,和一个增量版本5。而一个版本“5”只有主版本5,没有次版本和增量版本。限定版本用来标识里程碑构建:alpha和beta发布,限定版本通过连字符与主版本,次版本或增量版本隔离。例如,版本“1.3-beta-01”有一个主版本1,次版本3,和一个限定版本“beta-01”。


如果版本号与格式<主版本>.<次版本>.<增量版本>-<限定版本>相匹配,它就能被正确的比较;“1.2.3”将被评价成是一个比“1.0.2”更新的构件,这种比较基于主版本,次版本,和增量版本的数值。如果的版本发布号没有符合本节介绍的标准,那么的版本号只会根据字符串被比较;“1.0.1b”和“1.2.0b”会使用字符串比较。另外限定版本也是按照字符串比较排序的,如版本号“1.2.3-alpha-2”和“1.2.3-alpha-10”, Maven排序的结果是“alpha-10”比“alpha-2”更旧,如果使用“alpha-02”和“alpha-10”,这个问题就消除了。[/color]



[size=medium][color=red]21. 依赖范围[/color][/size]


<project>
  ...
  <dependencies>
    <dependency>
      <groupId>org.codehaus.xfire</groupId>
      <artifactId>xfire-java5</artifactId>
      <version>1.2.5</version>
    </dependency>
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>3.8.1</version>
      <scope>test</scope>
    </dependency>
    <dependency>
      <groupId>org.apache.geronimo.specs</groupId>
      <artifactId>geronimo-servlet_2.4_spec</artifactId>
      <version>1.0</version>
      <scope>provided</scope>
    </dependency>
  </dependencies>
  ...
</project>

[color=blue] 这里的第一个依赖是对于来自Codehaus的XFire SOAP库的编译范围(compile)依赖。如果你的项目在编译,测试,和运行中都依赖于一个类库,你就要使用这种依赖。第二种依赖是一个对于JUnit测试范围(test)的依赖。当你只有在测试的时候才引用类库的时候,你就要使用测试范围依赖。Example 9.3, “项目依赖”中最后一个依赖是对于由Apache Geronimo项目实现的Servlet 2.4 API的依赖。最后一项依赖的范围是已提供的(provided)依赖。当你的开发过程只有在编译和测试时需要一个类库,而该类库在运行的时候由容器提供,那么你就需要使用已提供范围的依赖。


附:[/color]


[color=red] [b]compile(编译范围):[/b][/color]


[color=blue] compile是默认的范围;如果没有提供一个范围,那该依赖的范围就是编译范围。编译范围依赖在所有的classpath中可用,同时它们也会被打包。[/color]


[color=red] [b]provided(已提供范围):[/b][/color]


[color=blue] provided依赖只有在当JDK或者一个容器已提供该依赖之后才使用。例如,如果你开发了一个web应用,你可能在编译classpath中需要可用的Servlet API来编译一个servlet,但是你不会想要在打包好的WAR中包含这个Servlet API;这个Servlet API JAR由你的应用服务器或者servlet容器提供。已提供范围的依赖在编译classpath(不是运行时)可用。它们不是传递性的,也不会被打包。[/color]


[color=red] [b]runtime(运行时范围):[/b][/color]


[color=blue] runtime依赖在运行和测试系统的时候需要,但在编译的时候不需要。比如,你可能在编译的时候只需要JDBC API JAR,而只有在运行的时候才需要JDBC驱动实现。[/color]


[color=red] [b]test(测试范围):[/b][/color]


[color=blue] test范围依赖 在一般的 编译和运行时都不需要,它们只有在测试编译和测试运行阶段可用。[/color]


[color=red] [b]system(系统范围):[/b][/color]


[color=blue] system范围依赖与provided类似,但是你必须显式的提供一个对于本地系统中JAR文件的路径。这么做是为了允许基于本地对象编译,而这些对象是系统类库的一部分。这样的构件应该是一直可用的,Maven也不会在仓库中去寻找它。如果你将一个依赖范围设置成系统范围,你必须同时提供一个systemPath元素。注意该范围是不推荐使用的(你应该一直尽量去从公共或定制的Maven仓库中引用依赖)。[/color]



[size=medium][color=red]22. 依赖管理[/color][/size]



[color=blue] 顶层POM中的依赖管理与在一个广泛共享的父POM中定义一个依赖是不同的。对初学者来说,所有依赖都会被继承。


如果mysql-connector-java在顶层父项目中被作为一个依赖列出,这个层次中的所有项目都将引用该依赖。为了不添加一些不必要的依赖,使用dependencyManagement能让你统一并集中化依赖版本的管理,而不用添加那些会被所有子项目继承的依赖。换句话说,dependencyManagement元素和一个环境变量一样,能让你在一个项目下面的任何地方声明一个依赖而不用指定一个版本号。[/color]