`
knightyzj
  • 浏览: 19663 次
  • 性别: Icon_minigender_1
  • 来自: 重庆
最近访客 更多访客>>
文章分类
社区版块
存档分类
最新评论

MSBuild的深入认识

阅读更多

最近在从事自动构造工作的过程中,对MSBuild本身有了一些更加深入的认识。MSBuild不仅仅是一个构造工具,应该称之为拥有相当强大扩展能力的自动化平台。

按照笔者现在的理解,MSBuild平台的主要涉及到三部分:执行引擎、构造工程、任务。其中最核心的就是执行引擎,它包括定义构造工程的规范,解释构造工程,执行“构造动作”;构造工程是用来描述构造任务的,大多数情况下我们使用MSBuild就是遵循规范,编写一个构造工程;MSBuild引擎执行的每一个“构造动作”就是通过任务实现的 ,任务就是MSBuild的扩展机制,通过编写新的任务就能够不断扩充MSBuild的执行能力。

所以这三部分分别代表了引擎、脚本和扩展能力。

1、构造工程(脚本文件)

先说说构造工程,只要通过Notepad打开任何一个VS2005(也就是支持CLR 2.0)下的C#工程(csproj)文件,就知道构造工程到底是怎么回事了。

如果说脚本,我们立刻想到的是VBScript或者JavaScript,构造工程内描述的内容,和常见的脚本语言的源文件之间还是有蛮大差距的,为什么也称之为“脚本”呢?因为笔者觉得没啥区别。脚本不就是纯文本形式保存,不经编译解释执行,可以实现一定逻辑分支的程序么?

再看构造工程,在构造工程中我们我们可以定义和使用变量(通过Property/PropertyGourp/Item/ItemGroup等元素),可以使用条件分支(通过Choose/When/Otherwise等元素)、能够在运行时给变量赋值(通过执行任务,获取其返回类型参数的方式)、能够定义执行块(通过Target元素,相当于函数)、能够进行异常处理(通过OnError元素)、还可以复用已有工程定义的内容(通过Import元素)。拥有这些能力和高级语言已经相差无几了,所以笔者认为构造工程不是描述性语言,而是脚本语言。

这里还需要强调一点的是,项目级元素(Property)可以在<PropertyGroup>元素下定义,也可以在构造过程中作为外部参数传入(具体参见《MSBuild命令行参考》)。这是一个非常有用的特性,一般编译时选择配置项(Debug或者Release)就是利用这个特性实现的。

有关构造工程的编写规范可以参考《MSBuild项目文件引用》。

2、执行引擎

接下来看执行引擎,通常我们使用下面的命令行开始执行构造:

MSBuild.exe <ProjectFile>

其中<ProjectFile>是前面提到的构造工程,也就是脚本文件,那么MSBuild.exe就应当是执行引擎了。

没错,不过看一下源代码就会发现MSBuild.exe非常简单,其实主要做的工作就是命令行解析、构造环境的准备(如生成日志记录模块准备一些全局变量),然后就是创建Microsoft.Build.BuildEngine.Engine类的实例,然后调用其BuildProjectFile方法来完成。所以真正的构造逻辑是在Microsoft.Build.Engine.dll中定义并且实现的。下面简单的代码就模拟了MSBuild.exe的工作。

具体的对象模型参见CLR类库参考中的《Microsoft.Build.Framework命名空间》和《Microsoft.Build.BuildEngine命令空间》。

笔者简单地分析了一下MSBuild.exeMicrosoft.Build.Engine.dll的源代码,MSBuild的构造过程大致如下:

a) 先创建一个构造请求(BuildRequest,构造请求是用来记录构造状态的数据结构),创建完毕之后将构造请求投递到请求队列中。

b) 在执行模块中,从请求队列中获取请求,然后开始处理。

c) 通过Project类加载构造工程,加载过程中检查是SulotionVC工程还是其它语言的工程。如果是Solution的话,生成一个临时的包装工程,逐一构造Solution中包含的工程;如果是VC工程的话,也生成一个包装工程,在这个工程中直接执行VCBuild任务来执行构造。否则直接通过XmlDocument加载项目文件,解析其中的元素,识别PropertyItemTarget之类元素。

d) 工程解析完毕后按照Target的顺序逐一执行。

e) 在执行Target的过程中先解析是否存在依赖的Target以及OnError子句(即产生错误时需要执行的Target)。

f) 先执行Target依赖的Target,然后通过TaskEngine执行本Target中的每一个任务。如果Target每一个任务都正确执行的话,那么执行下一个Target;否则执行错误处理的Target

g) 执行Task的过程就是实例化注册为Task的类,然后调用其Execute方法。

h) 所有Target执行完毕,则本次构造也执行完毕。

以上仅仅为了便于理解概念进行的描述,实际的构造过程可能是考虑到多CPU以及内联编译,内部逻辑相当复杂,很多地方应用了Proxy模式。

3、任务(Task

通过对执行引擎的描述可以发现执行引擎主要是维护执行流程以及记录执行流程中各类变量(PropertyItem),具体构造过程中的每一个动作,都是通过Task实现的。也就是说单靠Microsoft.Build.Engine.dll虽然可以加载并且解析构造工程,但是无法完成构造动作。之所以MSBuild能够完成编译、链接、创建目录、复制文件等一系列工作,都是因为在Microsoft.Build.Tasks.dll中实现了与之对应的一个个任务。具体请参考《MSBuild任务参考》以及CLR类库参考中的《Microsoft.Build.Task命名空间

通过观察MSBuild自带的这些常用任务,可以发现其中分为两类:一类从ToolTask继承,这类任务基本上就是直接调用外部二进制文件执行完成某个特定的动作,例如VCBuildExec等;另一类直接从Task继承,是通过内部代码逻辑完成特定动作,例如CopyMSBuild等。那些直接执行外部文件的任务虽然功能强大,但是有比较大的局限性,执行结果的反馈非常有限,通常只有ExitCode,很难获得其内部执行的更多信息,例如日志输出或者操作影响的结果。

大部分情况下我们只需要一个Exec任务就能够完成全部的构造动作,但是这样做的结果和我们写一个命令行的批处理文件没什么区别了。MSBuild平 台和命令行批处理最大不同在于,它是一个更紧密的工作环境,任务之间通过一系列自定义的全局参数互相协同工作,比较灵活并且移植性高;同时共享日志模块统 一输出执行过程中的各类信息,便于观察和分析。而批处理中各个命令之间几乎是完全孤立的,只能通过硬编码的方式进行协同,协作能力比较差。

举个最简单的例子,编译三个工程,然后复制编译结果到目标目录下。如果用批处理可能会写成这个样子:“编译工程1、复制编译结果、编译工程2、复制结果、编译工程3、复制结果”。而利用MSBuild就可以简化很多工作,例如“申明需要编译的工程(工程1、工程2、工程3...)、编译需要编译的工程、复制编译结果”。

顺便说一句,MSBuild这个任务比较有趣,在内部直接调用了构造引擎的方法(IBuildEngine2.BuildPrejectFilesInParallel)。这个例子又一次告诉我们这个世界上许多看似强大的东西,其实什么都没有干,只是因为他们手中掌握了有效的资源。-^o^-

除了基础的任务之外,任务还可以任意扩展,并且实现这种扩展非常方便。创建一个CLR 2.0以上的类库工程,编写实现了ITask接口的类,然后在构造工程中通过<UsingTask>元素注册任务就可以使用了。例如:

所以说MSBuild能够成为一个构造引擎,不是因为有个叫MSBuildExe文件,也不是因为脚本文件被称之为构造工程,而是因为与之配套的Microsoft.Build.Task.dll中主要实现了主要是和构造相关的任务。换句话说如果提供和测试相关的任务库的话,MSBuild也就是一个自动测试的平台。

总之,MSBuild本身更趋向于一个自动化执行平台,可以根据需求编写不同的脚本文件来满足不同的应用,当现有能力无法满足时,通过编写新的任务进行扩展。不仅限于构造,自动安装、自动测试等都可以依赖这个平台来实现。

分享到:
评论

相关推荐

    MSBuild 12.0 14.0 17.0

    MSBuild是Microsoft构建引擎的缩写,它是微软提供的一款用于构建.NET应用程序的工具。自2005年随Visual Studio .NET 2005首次发布以来,MSBuild已经成为.NET开发不可或缺的一部分,经历了多个版本的迭代,包括...

    MSBuild V12编译器、BuildTools_Full

    MSBuild V12是Microsoft的一款构建工具,它与Visual Studio紧密相关,用于自动化构建.NET Framework应用程序。...通过深入学习MSBuild的工作原理和使用技巧,开发者可以更高效地管理和维护他们的项目构建流程。

    微软 MSBuild 详细介绍

    【MSBuild 详细介绍】 MSBuild(Microsoft Build Engine)是微软推出的一个构建工具,它负责编译和构建应用程序...通过深入理解MSBuild的工作机制和XML项目文件格式,开发者可以更有效地优化和自动化他们的构建过程。

    MSbuild使用MSBuild实现完整daily_build流程

    ### 使用MSBuild实现完整daily_build流程 #### 一、MSBuild概述 MSBuild(Microsoft Build Engine)是微软推出的一种基于XML的平台独立的构建工具,主要用于.NET应用的自动化构建过程。它不仅支持C#和Visual Basic...

    sonar-scanner-msbuild-5.3.1.36242-net46.zip

    本文将深入探讨SonarScanner MSBuild的工作原理以及压缩包内包含的文件及其作用。 SonarScanner MSBuild的主要功能是在MSBuild构建过程中插入代码分析步骤,它能够自动检测源代码中的潜在问题,如bug、漏洞、复杂性...

    VisualStudio的生成系统MSBuild.zip

    MSBuild 是 Microsoft Build Engine 的缩写,MSBuild 是 Visual Studio 的生成系统。MSBuild 在如何处理和生成软件方面是完全透明的,使开发人员能够在未安装 Visual Studio 的生成实验室环境中组织和生成产品。 ...

    visial studio2019缺少的MsBuild文件

    在本文中,我们将深入探讨MsBuild,以及如何解决在Visual Studio 2019中缺失MsBuild文件的问题。 首先,我们要理解MsBuild的作用。MsBuild是一个基于XML的构建系统,用于管理和执行构建过程。它负责解析项目的构建...

    Inside the Microsoft Build Engine: Using MSBuild and Team Foundation Build

    ### 使用MSBuild与Team Foundation Build:深入微软构建引擎 #### 一、概述 《Inside the Microsoft Build Engine: Using MSBuild and Team Foundation Build》是一本由Sayed Hashimi和William Bartholomew共同...

    msbuild.hpi 1.12

    msbuild.hpi 1.12 好不容易找到的

    MSBuild 说明文档

    MSBuild 文件格式规范详解 MSBuild 文件格式是微软公司推出的一个基于 XML 的构建自动化工具,可以让项目作者声明性地描述三个基本要素:WHAT TO BUILD、HOW TO BUILD 和 WITH WHAT PROPERTIES。该文件格式旨在让...

    MSBuild 经典教程 2

    本教程将从 MSBuild 的基础知识开始,逐步深入到高级特性,帮助读者掌握 MSBuild 的使用和应用。 MSBuild 文件格式 MSBuild 文件格式是 MSBuild 的核心组件之一,它允许项目作者声明项目的结构、依赖关系和构建...

    批处理调用MSBuild自动生成DLL资源文件

    使用批处理文件调用MSBuild自动生成DLL资源文件,需要将需要生成dll资源文件的文件名添加在一个txt文档中以便批处理文件调用。

    MSBuild参数说明

    涵盖了MSBuild的命令行参数,是编写MSBuild自动执行脚本的参考。

    MSBuild安装软件

    MSBuild安装软件

    CruiseControl.NET + msbuild的安装与配置方法

    CruiseControl.NET + msbuild的安装与配置方法 CruiseControl.NET是一款自动化的持续集成工具,可以与msbuild集成,以实现项目的自动编译和监视。本文将详细介绍CruiseControl.NET + msbuild的安装与配置方法。 ...

    MSBuild.rar

    MSBuild是微软开发的一种构建工具,它用于自动化构建.NET Framework应用程序。这个名为"MSBuild.rar"的压缩包可能包含了与MSBuild相关的配置文件,特别是针对x64平台的Microsoft.Cpp.x64.user的设置。在.NET开发环境...

    使用MSBuild实现完整daily build流程

    "使用MSBuild实现完整daily build流程" 基于MSBuild的daily build流程是指使用MSBuild工具来自动化整个构建过程,使得开发团队可以更方便地管理和维护构建流程。MSBuild是一款强大且灵活的构建工具,可以直接读取...

    sonar-msbuild-runner.zip

    sonar-msbuild-runner 是 Sonar 提供对微软 MSBuild 的集成支持。 标签:sonar

Global site tag (gtag.js) - Google Analytics