6

我有一个我部分拥有的代码生成器工具,现在 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

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

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

4

5 回答 5

9

单一目标框架上,我只使用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>

在我使用的多目标框架上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>

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

于 2019-02-01T11:46:48.243 回答
5

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

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

于 2017-10-11T04:08:04.257 回答
3

这里的答案非常有用,但我的情况需要一个通用目标,可以由具有单个或多个 TargetFrameworks 的项目调用。我想出了一个 Singleton 目标,它在构建之前每个项目运行一次,无论 TargetFrameworks 的数量如何。所有这些情况的单次运行:

  • <TargetFrameworks>netstandard2.0;net4.6</TargetFrameworks>
  • <TargetFrameworks>netstandard2.0</TargetFrameworks>
  • <TargetFramework>netstandard2.0</TargetFramework>

此解决方案的关键是设置一个属性(在下面的示例中名为 IsCrossTargetingProject),该属性允许各个构建知道它们是否由 DispatchToInnerBuilds 启动,因此应该只在外部构建中运行。

我本可以举一个例子,所以我决定在这里为遇到这种情况的其他人发布它。

例子:

<Project>
    <PropertyGroup>
        <IsCrossTargetingProject Condition="$(TargetFrameworks) != ''">true</IsCrossTargetingProject>
    </PropertyGroup>

    <!--Only used to display properties to understand what the Singleton Target is doing.  Not required-->
    <Target Name="DebuggingDisplayProperties" BeforeTargets="DispatchToInnerBuilds;BeforeBuild">
        <Message Text="      ######################################################## Display Properties for $(MSBuildProjectName) $(TargetFramework)." Importance="high" />
        <Message Text="      IsCrossTargetingBuild: '$(IsCrossTargetingBuild)'" Importance="high" />
        <Message Text="      IsCrossTargetingProject: '$(IsCrossTargetingProject)'" Importance="high" />        
        <Message Text="      Compare => '$(IsCrossTargetingProject)' == '$(IsCrossTargetingBuild)'" Importance="high" />        
    </Target>
    
    <Target Name="SingletonTarget" Condition="'$(IsCrossTargetingProject)' == '$(IsCrossTargetingBuild)'" BeforeTargets="DispatchToInnerBuilds;BeforeBuild" >
        <!--Begin sample target body--> 
        <PropertyGroup Condition="'$(IsCrossTargetingBuild)' == ''">
            <CrossTargetMessage>$(TargetFramework)</CrossTargetMessage>
        </PropertyGroup>
        <PropertyGroup Condition="'$(IsCrossTargetingBuild)' == 'true'">
            <CrossTargetMessage>Multi-Targeted</CrossTargetMessage>
        </PropertyGroup>
        <Message Text="      ######################################################## Singleton Target for $(MSBuildProjectName) - $(CrossTargetMessage)." Importance="high" />
        <!--End sample target body--> 
    </Target>
</Project>

我将它放在一个单独的文件中,并在我的主项目文件中使用以下行将其导入:

<Import Project="..\Common\Singleton.targets" />
于 2021-12-14T23:47:39.493 回答
1
Condition="'$(TargetFrameworks)' == '' OR $(TargetFrameworks.EndsWith($(TargetFramework)))" 
于 2020-01-18T22:37:02.837 回答
-1

另一种方法是GenerateNuspec在您想要更改 nuget 文件时使用。

打包时会找到一次 Foo 输出。

于 2019-11-21T02:25:54.423 回答