MSbuild构建订单问题 - 首先构建预先构建步骤或首先依赖项目

时间:2012-08-27 15:48:47

标签: c# msbuild

我有一个项目A取决于项目B.项目A有一些预构建任务,它们依赖于项目B中的一些生成文件。当我在Visual Studio中构建时,没有问题。但是在使用MSBuild.exe时,则存在问题,因为构建顺序为:

  • A的预构建步骤< - 因为B尚未编译而失败
  • B编译< - 期望首先执行
  • A已编译

是否是使用MSBuild的预期行为? 有没有办法告诉MSBuild在A的预建步骤之前先做B?

我正在使用VS2010 C#和C ++ / CLI。我不认为是否提供额外的信息,但这是如何被称为:

Running process (C:\Windows\Microsoft.NET\Framework\v4.0.30319\MSBUILD.exe "..\..\..\dev\build\MyProj.sln" /t:Clean /p:Configuration=Release;Platform=Win32)

4 个答案:

答案 0 :(得分:11)

简短回答

删除构建事件,并在项目中添加以下内容(.csproj):

<Target Name="AfterResolveReferences">
  <Exec Command="echo helloworld" />
</Target>

更多信息

您可以在此处阅读有关自定义构建过程的信息:http://msdn.microsoft.com/en-us/library/ms366724%28v=vs.110%29.aspx

在ResolveReferences之前触发Before Build事件,因此如果在项目的BeforeBuild事件发生之前尚未构建您引用的项目,则BeforeBuild事件将失败。

要解决此问题,您应该使用不同的入口点来自定义构建过程。在上面的示例中,我使用了AfterResolveReferences,因为这将确保您引用的所有项目都已构建。

答案 1 :(得分:6)

杰克的回答似乎有效,但我不喜欢的是VS UI中原生支持编辑.csproj的支持, 除了尴尬的“卸载项目,编辑项目(在此期间你不能像往常一样点击你的文件),重新加载项目”模型。 我想要的是在构建依赖项目之后触发预构建事件,并且在VS中和MSBuild中的工作相同。 在努力解决这个问题后,我找到了一个适用于MSBuild 4.0的解决方案。

无论我尝试什么,我都无法在依赖项目完成构建后更改PreBuildEvents目标。所以我做了什么 是禁用PreBuildEvents目标,并创建一个在适当的时间运行的私有自定义PreBuildEvents目标:

<ItemGroup>
  <ProjectReference Include="..\YourProjectPath\YourProject.csproj">
    <ReferenceOutputAssembly>false</ReferenceOutputAssembly>
  </ProjectReference>
</ItemGroup>
<Target Name="PreBuildEvent" AfterTargets="" BeforeTargets="" />
<Target Name="BastardPreBuildEvent" AfterTargets="ResolveReferences" BeforeTargets="CoreResGen">
  <Exec Command="$(PreBuildEvent)" />
</Target>

答案 2 :(得分:3)

这是一个相当古老的问题,但我找到了这个简单的解决方案(也可能适用于你) 只需添加

<PreBuildEventDependsOn>ResolveReferences</PreBuildEventDependsOn>

之前

<PreBuildEvent>

不幸的是,这必须在编辑器中完成,VS编辑器不支持它(版本15.6.2)。

答案 3 :(得分:0)

这可能很旧但我最近遇到了同样的问题。杰克的回答是我试图使用的,但ResolveReferences目标是在Visual Studio中加载项目时调用的,这对我来说是糟糕,因为我在目标期间做了一个相当长的过程,我只想要在构建之前做。

我最终使用的解决方案类似于VeeTheSecond,但涉及的内容较少:

<Target Name="BeforeBuild" DependsOnTargets="ResolveReferences">
    ... exec something using a dependant project's output exe ...
</Target>

这将强制BeforeBuild目标等待所有依赖项目实际构建完成,以便exec命令可以使用依赖项目的输出。