通常在一个工程中构建多个项目间会有关联,尤其是它们都依赖一个项目时可以很容易的更新项目
在一个工程中每个子项目都会有自己的源代码目录、生成各自的jar包当执行 package 时.
一个项目通过申明一个 Project 类型的懒值来定义,例如:
lazy val util = project
lazy val core = project
这个变量值名称将被用来当做 Project Id 和项目的根目录名称,这个ID将用来在命令行中引用项目,利用方法in 可以修改默认的项目根目录。例如, 以下是更加明确的申明项目:
lazy val util = project.in(file("util"))
lazy val core = project in file("core")
在一个工程中一个项目完全可能依赖另一个项目,经常有两种依赖方式:聚合和classpath
项目聚合
聚合的意思是当运行一个任务在一个项目中,其通过聚合方式依赖的项目也会执行,例如:
lazy val root = (project in file(".")).
aggregate(util, core)
lazy val util = project
lazy val core = project
在上面的例子中,root项目聚合了项目util和core,编译root项目将看到三个项目被编译。
在项目聚合过程中,以root项目为例,是可以控制任务维度作用域的配置的,例如,可以控制在执行update这个任务时不进行聚合:
lazy val root = (project in file(".")).
aggregate(util, core).
settings(
aggregate in update := false
)
[...]
aggregate in update 表示在update这个任务维度作用域中aggregate的配置。(具体可参考作用域章节)
注意:在聚合过程中聚合是并行处理的,所以被聚合的项目时没有先后顺序的
classpath 依赖
在源代码层级一个项目可能会依赖另一个项目,可以通过 dependsOn 方法添加依赖关系,例如,core项目需要在classpath中指定util项目,可以这样定义core项目:
lazy val core = project.dependsOn(util)
这样就可以在core项目中调用util项目的方法, 当编译的时候会有编译顺序,util必须在core之前编译, 如果依赖多个项目,可以给dependsOn方法指定多个参数,例如,dependsOn(bar, baz)
配置维度作用域的 classpath 依赖
foo dependsOn(bar) 表示foo在Compile这个配置维度作用域下依赖在配置维度作用域为Compile下的bar项目,确切的写法应该为:dependsOn(bar % "compile->compile"), 在"compile->compile"中的->表示项目间的依赖关系,所以"test->compile"可以表示为在Test配置维度作用域下的foo依赖Compile配置维度作用域下的bar
省略"->config"这部分隐含意思为"->compile", 所以dependsOn(bar % "test")意思是在Test配置维度作用域下的foo依赖Compile配置维度作用域下的bar
可以声明为"test->test",意思为在Test配置维度作用域下项目依赖,例如,在src/test/scala目录的公共类库可以在src/test/scala源代码中使用。
针对一个依赖关系可以配置多个配置维度的作用域,用分号分隔,例如:dependsOn(bar % "test->test;compile->compile")
如果一个项目没有在工程根目录下定义,sbt将创建一个聚合整个工程子项目的默认项目。
由于项目hello-foo 定义了base = file("foo"),项目目录为子目录foo, 源代码可以直接放到foo目录中,类似foo/Foo.scala或放到src/main/scala,如果是采用sbt工程构建标准目录,foo目录下还包括工程构建定义文件。
在foo目录下的任何.sbt文件将被合并在一起,并且定义的配置项属于项目维度hello-foo作用域。
在hello工程中允许子项目配置不同的版本,可以在配置文件:hello/build.sbt, hello/foo/build.sbt, 和hello/bar/build.sbt中配置不同版本,现在在交互模式下执行show version将看到如下信息:
> show version
[info] hello-foo/*:version
[info] 0.7
[info] hello-bar/*:version
[info] 0.9
[info] hello/*:version
[info] 0.5
hello-foo/:version被定义在hello/foo/build.sbt中 hello-bar/:version被定义在hello/bar/build.sbt中 hello/*:version被定义在hello/build.sbt中
每个version是不同的项目维度作用域,但是这三个build.sbt部分配置是相同的
每个项目的配置都在自己项目目录下的 .sbt 文件中配置,其实对于上述配置有更简单的方法,那就是配置到 .scala 文件中,列举出项目和对应的项目根目录
你会发现将多个项目按顺序定义在 .scala 配置文件中会比单独定义在各自的目录下更加清晰,不过这个自己决定。
不允许定义一个子目录为project的目录,foo/project/Build.scala将会被忽略
在sbt的交互模式下,使用命令projects 列举出该工程的所有项目,命令project
.sbt 配置文件中的定义在多个 .sbt 中是相互不可见的,为了是多个.sbt间复用配置,可以在根目录的project目录下定义一个或多个 .scala 的文件,这个目录其实也是一个sbt工程,只不过这个的作用是工程构建。
例如:
import sbt._
import Keys._
object Common {
def text = "org.example"
}
在.sbt中可以直接调用:
organization := Common.text