首先,Target 节点上的 Inputs 和 Outputs 属性用于增量构建。它们需要具有相同数量的条目,以便 msbuild 了解在构建时应该过滤哪些项目。MSBuild 检查输出是否已经存在并且是最新的,如果是,则从输入列表中过滤匹配的输入项。如果您不关心增量构建,则可以完全跳过此机制。如果输入和输出不匹配(或不存在),msbuild 将始终使用所有项目执行目标,因为它无法决定哪些项目导致哪些输出。
其次,这些属性所期望的是项目列表。这不必是一个列表,它可以是任意列表。因此,像这样扩展您的示例是完全公平的:
<Target Name="Backup" Inputs="@(Compile);@(Compile2)"
Outputs="@(Compile->'$(BackupFolder)%(Identity).bak');@(Compile2->'$(BackupFolder)%(Identity).bak')">
<Copy SourceFiles="@(Compile)" DestinationFiles=
"@(Compile->'$(BackupFolder)%(Identity).bak')" />
<Copy SourceFiles="@(Compile2)" DestinationFiles=
"@(Compile2->'$(BackupFolder)%(Identity).bak')" />
</Target>
但是您想将相同的项目复制到不同的备份文件夹,对吗?所以这样的事情应该做:
<Target Name="Backup">
<Copy SourceFiles="@(Compile)" DestinationFiles=
"@(Compile->'$(BackupFolder)%(Identity).bak')" />
<Copy SourceFiles="@(Compile)" DestinationFiles=
"@(Compile->'$(BackupFolder2)%(Identity).bak')" />
</Target>
使用两个备份文件夹,一个项目实际上可能在一个文件夹中已经是最新的,但在另一个文件夹中丢失。您可以将一个定义为“主”备份文件夹,并告诉 MSBuild 将其用作增量构建的参考。
编辑:对于两个位置的增量构建,可能最简单的解决方案是组合两个目标,两者都是增量构建:
<Target Name="Backup" DependsOnTargets="_Backup1;_Backup2">
</Target>
<Target Name="_Backup1" Inputs="@(Compile)"
Outputs="@(Compile->'$(BackupFolder)%(Identity).bak')">
<Copy SourceFiles="@(Compile)" DestinationFiles=
"@(Compile->'$(BackupFolder)%(Identity).bak')" />
</Target>
<Target Name="_Backup2" Inputs="@(Compile)"
Outputs="@(Compile->'$(BackupFolder2)%(Identity).bak')">
<Copy SourceFiles="@(Compile)" DestinationFiles=
"@(Compile->'$(BackupFolder2)%(Identity).bak')" />
</Target>