在 TargetFrameworks 标记中每个框架运行一次的目标之前,如何制作只运行一次而不是一次的 MSBuild 目标?

Jer*_*ton 3 c# csproj visual-studio-2017 msbuild-15

我有一个我部分拥有的代码生成器工具,现在 csproj 文件可以在其中列出多个目标框架并构建所有这些框架,我试图弄清楚如何使 MSBuild 目标只进行一次代码生成运行构建,无论列出了多少个目标框架,并让每个目标框架的编译等待代码生成完成。

我目前以 TargetFramework 的特定值为条件。例如,Condition="'netstandard2.0' == '$(TargetFramework)'"

这避免了同时为每个目标框架启动代码生成工具,然后在进程尝试创建/更新相同文件时出现访问被拒绝错误。

但是,其他目标框架会在不等待代码生成完成的情况下直接尝试编译,并且在没有该代码的情况下失败。

我希望每次构建项目时只生成一次代码,并且每个目标框架的编译只在完成后开始。

它需要在每次构建运行时运行,以防输入发生变化并生成不同的代码。

注意:目前我忽略了一种情况,其中#if FrameworkSpecificDefine用于代码生成器的输入代码不同,因此不同的目标框架会导致代码生成器的输出不同。目前,我正在考虑代码生成器的输出在所有目标框架中都相同且有效。

更新:在寻找在 MSBuild 拆分为 TargetFramework 特定构建之前发生的目标之后,我可以在之前挂钩构建,我在 VS 的详细构建输出中看到了这一点:

1>Target _SetBuildInnerTarget:
1>Target _ComputeTargetFrameworkItems:
1>Target DispatchToInnerBuilds:
1>  Using "MSBuild" task from assembly "Microsoft.Build.Tasks.Core, Version=15.1.0.0, Culture=neutral, PublicKeyToken=b03f5f7f11d50a3a".
1>  Task "MSBuild"
1>    Additional Properties for project "ProjectA.csproj":
1>      TargetFramework=netstandard2.0
1>    Additional Properties for project "ProjectA.csproj":
1>      TargetFramework=net47
Run Code Online (Sandbox Code Playgroud)

然后我将我的目标设置为BeforeTargets="DispatchToInnerBuilds",它在专门设置 TargetFramework 的个人构建之前运行,并且似乎完全满足我的需求。

(添加到BuildDependsOn属性似乎不再起作用:添加在 Visual Studio 2017 RC 中构建 .NET Core 项目后运行的 msbuild 任务;我怀疑 Microsoft.Common.targets 稍后会以新的 csproj 格式进行评估,并且您在项目文件中所做的任何附加到属性都会被 Microsoft.Common.targets 覆盖。)

Dom*_*nas 6

单一目标框架上,我只使用BeforeTargets="PreBuildEvent"

<Project Sdk="Microsoft.NET.Sdk">
  <PropertyGroup>
    <TargetFramework>netstandard2.0</TargetFramework>
  </PropertyGroup>
  <Target Name="GenerateVersionInfo" BeforeTargets="PreBuildEvent">
    <Exec Command="your custom command" />
  </Target>
</Project>
Run Code Online (Sandbox Code Playgroud)

在我使用的多目标框架上BeforeTargets="DispatchToInnerBuilds"

<Project Sdk="Microsoft.NET.Sdk">
  <PropertyGroup>
    <TargetFrameworks>netstandard2.0;net461</TargetFrameworks>
  </PropertyGroup>
  <Target Name="GenerateVersionInfo" BeforeTargets="DispatchToInnerBuilds">
    <Exec Command="your custom command" />
  </Target>
</Project>
Run Code Online (Sandbox Code Playgroud)

所以我的自定义命令在每次构建之前只执行一次。如果您使用InitialTargets,则该命令的执行次数将不止一次!例如,如果您保存项目!

  • `DispatchToInnerBuilds` 似乎只适用于具有 `TargetFrameworks` 的项目(尽管它可能仍然只包含一项),并且仅在从 Visual Studio 运行构建时有效。从命令行,“dotnetpublish”需要指定单个框架,并且“DispatchToInnerBuilds”不会触发。 (3认同)

Mar*_*ich 5

虽然某些目标仅在内部构建中运行,例如当您使用 时BeforeTargets="BeforeBuild",外部构建还定义了IsCrossTargetingBuild变量来指示当前运行的构建是分派到内部构建的外部构建,并且是条件目标的首选方式。

因此,您可以调整目标,Condition="'$(IsCrossTargetingBuild)' == 'true'"以确保目标仅针对外部构建运行。

  • 您能详细说明一下您的解决方案吗?目标仍然是 `BeforeTargets="BeforeBuild"` 加条件,还是 `BeforeTargets="DispatchToInnerBuilds"` 加条件或任何其他组合?如果项目是多目标的,但仅为一个目标框架调用构建,该怎么办?您能否提出一个无论构建是从 IDE 还是 CLI 启动都有效的解决方案? (2认同)