【问题标题】:How can I build multiple configurations in parallel?如何并行构建多个配置?
【发布时间】:2012-10-20 12:19:11
【问题描述】:

我有一个包含十二个解决方案配置的 Visual Studio 2012 解决方案。每个解决方案配置都是独立的(即每个配置的输出完全不相关)。

问题:如何一步构建所有十二个配置,即通过在命令行上运行一个 MSBuild 命令,以及如何并行构建配置?

例如,如果只有两种配置,Release/AnyCPU 和 Debug/AnyCPU,我希望这两种配置同时并行构建。

为了完整起见,以下是我尝试过的;我还没有解决这个问题的方法。


为了一次构建所有项目,我创建了一个新项目文件,其中包含一个构建目标,该目标在每个配置的解决方案文件上运行 MSBuild 任务:

<Target Name="Build">
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Release;Platform=Win32"     Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Release;Platform=x64"       Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Release;Platform=ARM"       Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Release(ZW);Platform=Win32" Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Release(ZW);Platform=x64"   Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Release(ZW);Platform=ARM"   Targets="$(BuildCommand)" />

  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Debug;Platform=Win32"       Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Debug;Platform=x64"         Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Debug;Platform=ARM"         Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Debug(ZW);Platform=Win32"   Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Debug(ZW);Platform=x64"     Targets="$(BuildCommand)" />
  <MSBuild Projects="cxxreflect.sln" Properties="SolutionDir=$(MSBuildProjectDirectory)\;Configuration=Debug(ZW);Platform=ARM"     Targets="$(BuildCommand)" />
</Target>

这很好用,除了每个 MSBuild 任务都是按顺序调用的,所以没有并行性(是的,每个配置构建中都有并行性,但我真的很想获得跨配置的并行性构建)。

为了让配置并行构建,我尝试利用 MSBuild 任务的 BuildInParallel 属性。我编写了一个预构建任务,为每个配置生成项目文件,然后尝试并行构建所有这些生成的项目:

<Target Name="PreBuild" Outputs="%(ProjectConfiguration.Identity)" Returns="%(BuildProject.Identity)">
  <Message Text="Cloning Solution for Configuration:  %(ProjectConfiguration.Identity)" />
  <PropertyGroup>
    <BuildProjectPath>$(IntPath)\%(ProjectConfiguration.Platform)\%(ProjectConfiguration.Configuration)\build.proj</BuildProjectPath>
  </PropertyGroup>
  <ItemGroup>
    <BuildProject Include="$(BuildProjectPath)" />
  </ItemGroup>
  <MakeDir Directories="$(IntPath)\%(ProjectConfiguration.Platform)\%(ProjectConfiguration.Configuration)" />
  <WriteLinesToFile
    File="$(BuildProjectPath)"
    Lines="&lt;?xml version='1.0' encoding='utf-8'?&gt;
&lt;Project DefaultTargets='Build' ToolsVersion='4.0' xmlns='http://schemas.microsoft.com/developer/msbuild/2003'>
  &lt;Target Name='Build'&gt;
    &lt;MSBuild
      Projects='$(SlnPath)'
      Properties='
        SolutionDir=$(MSBuildProjectDirectory)\%3b
        Configuration=%(ProjectConfiguration.Configuration)%3b
        Platform=%(ProjectConfiguration.Platform)
      '
      Targets='$(BuildCommand)'
    /&gt;
  &lt;/Target&gt;
&lt;/Project&gt;"
    Overwrite="true" />

</Target>
<Target Name="Build" DependsOnTargets="PreBuild">
  <Message Text="%(BuildProject.Identity)" />
  <MSBuild Projects="%(BuildProject.Identity)" Properties="BuildCommand=$(BuildCommand)" BuildInParallel="true" />
</Target>

不幸的是,虽然这确实构建了所有十二种配置,但它是按顺序构建的。我的猜测是,当 MSBuild 对要构建的项目集执行依赖关系分析时,它会识别出它们都依赖于同一个解决方案文件,因此它决定它们不能并行构建。 (这只是一个猜测;我可能完全错了。我对 MSBuild 知之甚少。)

我也在构建时使用 /m 开关。

【问题讨论】:

  • 至于为什么这很重要:如果我可以让构建并行运行,我可以将构建时间减少至少 37%。 (我通过自己从命令行运行 12 个 msbuild 实例对此进行了测试,发现构建性能提高了 37%——4:09 与 6:40。这样做的缺点是它破坏了构建的统一日志记录; 每个构建都会发出自己的日志。)

标签: msbuild


【解决方案1】:

利用 MSBuild 任务的 BuildInParallel 选项,在一次调用中传递所有项目。示例here 给出了基本方法:

<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
    <ItemGroup>
        <ProjectToBuild Include="a1.sln">
            <Properties>Configuration=Debug</Properties>
        </ProjectToBuild>
        <ProjectToBuild Include="a1.sln">
            <Properties>Configuration=Release</Properties>
        </ProjectToBuild>
    </ItemGroup>
    <Target Name="Build">
        <MSBuild Projects="@(ProjectToBuild)" BuildInParallel="true" />
    </Target>
</Project>

【讨论】:

  • 如果两个配置都继续执行另一个任务,是否可以将它嵌套在每个配置下,以便每个配置的下一个任务也并行执行?例如,它不会等待发布和调试完成,然后执行任务 A 进行发布,任务 A 进行调试,而是在调试完成后立即启动 A 进行调试,并在发布完成后立即启动 A 进行发布?
  • 如果我正确理解您的问题:这种方法将两个构建作为完全独立的操作开始。每个都将尽可能快地完成,两者之间没有同步。想象一下,您刚刚启动了两个控制台窗口,并在每个窗口中输入了 msbuild。
  • 所以如果您希望仅在构建这些命令时才执行另一个命令,那么这是不可能的?
  • 当然有可能……只需在 Build 目标中添加更多内容。
  • 但是最好将这样的查询作为一个新问题提出,而不是在不相关的问题中发表评论。
猜你喜欢
  • 2011-06-30
  • 2013-08-13
  • 1970-01-01
  • 2020-07-24
  • 1970-01-01
  • 2019-03-07
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多