我是 MSBuild 的新手,想尝试一下它,但我就是不明白为什么这不起作用。
所以我的解决方案有两个项目:“Model”和“BuildTasks”。 BuildTasks 只有一个类:
using Microsoft.Build.Utilities;
namespace BuildTasks
{
public class Test : Task
{
public override bool Execute()
{
Log.LogMessage( "FASDfasdf" );
return true;
}
}
}
然后在 Model.csproj 中我添加了以下内容:
<UsingTask TaskName="BuildTasks.Test" AssemblyFile="$(SolutionDir)src\BuildTasks\bin\BuildTasks.dll" />
<Target Name="AfterBuild">
<Test />
</Target>
我已经设置了构建顺序,因此“BuildTasks”在“Model”之前构建。但是当我尝试构建模型时,我收到此错误:
“BuildTasks.Test”任务无法 从程序集中加载 C:\WIP\TestSolution\src\BuildTasks in\BuildTasks.dll。 无法加载文件或程序集 '文件:///C:\WIP\TestSolution\src\BuildTasks in\BuildTasks.dll' 或其依赖项之一。系统 找不到指定的文件。 确认
声明是正确的,并且 程序集及其所有依赖项是 可用。
这个文件肯定存在,为什么MSBuild找不到它?
我什至尝试硬编码“C:\WIP\TestSolution”来代替“$(SolutionDir)”并得到相同的错误。但是,如果我将该 .dll 复制到桌面并将路径硬编码到桌面,它确实可以工作,但我不明白为什么。
编辑:我的路径没有错误。我修改了 BuildTasks 的调试/发布版本,将 .dll 输出到 bin 文件夹,因为我不希望调试/发布有不同的路径。
我们尝试了这一点,发现您必须将UsingTask放在项目文件的顶部(并且所有路径都正确)。然而,一旦完成并且任务加载,它就只能工作一次。之后,构建开始失败,因为它无法复制任务所在的 DLL。我们实际上正在运行一个构建后任务,该任务位于我们正在构建的同一程序集/项目中。
我们为解决此问题所做的就是在单独的 MSBuild 文件上启动单独的 MSBuild 进程来运行构建后任务。这样,DLL 在构建并复制到 bin 目录之后才会被加载。
<Target Name="AfterBuild">
<Exec Command="$(MSBuildBinPath)\MSBuild.exe
"$(MSBuildProjectDirectory)\PostBuild.msbuild"
/property:SomeProperty=$(SomeProperty)" />
</Target>
请注意,您可以在命令行上将属性传递到此子构建任务中。
PostBuild.msbuild 看起来像这样:
<?xml version="1.0" encoding="utf-8"?>
<Project DefaultTargets="PostBuild" xmlns="http://schemas.microsoft.com/developer/msbuild/2003" ToolsVersion="4.0">
<UsingTask TaskName="PostBuild" AssemblyFile="$(MSBuildProjectDirectory)\bin\AssemblyThatJustBuiltAndContainsBuildTask.dll" />
<PropertyGroup>
<SomeProperty>SomePropertyDefaultValue</SomeProperty>
</PropertyGroup>
<Target Name="PostBuild">
<MyPostBuildTask SomeProperty="$(SomeProperty)" />
</Target>
</Project>
如果所有自定义任务都继承自AppDomainIsolatedTask,则可以避免自定义任务DLL的锁定 如果您在启动 Visual Studio 之前设置环境变量
MSBUILDDISABLENODEREUSE = 1
。
具有自定义构建任务的程序集被 devenv.exe 和 msbuild.exe 锁定。
您可以使用
MSBUILDDISABLENODEREUSE = 1
使 msbuild.exe 进程消失,但如果您的自定义任务继承自 Task,devenv.exe 仍会偶尔锁定自定义任务程序集。
另一方面,如果您仅继承 AppDomainIsolatedTask 并且不设置
MSBUILDDISABLENODEREUSE
,空闲的 MSBuild 进程仍将锁定程序集。
Slace说得对。您的装配路径很可能是错误的。大概应该是:
<UsingTask
TaskName="BuildTasks.Test"
AssemblyFile="$(SolutionDir)src\BuildTasks\bin\$(Configuration)\BuildTasks.dll" />
<Target Name="AfterBuild">
<Test />
</Target>
禁用 MSBuild 节点重用也将解决此问题:
/nr:false
选项。MSBUILDDISABLENODEREUSE
环境变量设置为 1
。您可以尝试使用fuslogvw来诊断问题,我想我不清楚它是否已经发展到那么远了......
在我们的例子中,将 PlaformTarget 添加到项目文件 (.csproj) 中解决了问题。
你确定你写了你的路径吗?不是应该在
bin\Configuration Type\BuildTasks.dll
吗?
我找到了这个链接:http://bartdesmet.net/blogs/bart/archive/2008/02/15/the-custom-msbuild-task-cookbook.aspx(存档副本)在开始时非常有帮助编写 MSBuild 任务。