本文告诉大家如何在项目文件通过不同的条件使用不同的方法运行
本文是 手把手教你写 Roslyn 修改编译 的文章,在阅读本文之前,希望已经知道了大多数关于 msbuild 的知识
为了告诉大家如何使用判断,需要创建一个简单的程序来写,这里创建一个 dotnet core 控制台项目,如果还没安装 dotnet core 可以到 dotnet sdk 2.1.300 winx64-CSDN下载
从 VisualStudio 安装文件夹打开开发人员工具命令行,打开这个是因为如果没有将 msbuild
加入到环境变量,就需要使用这个方法来调用 msbuild
调用 msbuild 的原因是为了编译可以看到输出。
在 Target
如果用 Message
的输出,除了设置为错误,其他的输出在 VisualStudio 的输出是无法看到的,只能通过 msbuild
才可以看到。
打开开发人员工具,先进入到刚才创建的项目所在的文件夹,然后执行msbuild
就可以看到输出
C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi>msbuild
用于 .NET Framework 的 Microsoft (R) 生成引擎版本 15.7.180.61344
版权所有(C) Microsoft Corporation。保留所有权利。
生成启动时间为 2018/8/3 20:05:47。
节点 1 上的项目“C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi\RaskerYadeacorLalmi.csproj”(默认目标)。
GenerateTargetFrameworkMonikerAttribute:
正在跳过目标“GenerateTargetFrameworkMonikerAttribute”,因为所有输出文件相对于输入文件而言都是最新的。
CoreGenerateAssemblyInfo:
正在跳过目标“CoreGenerateAssemblyInfo”,因为所有输出文件相对于输入文件而言都是最新的。
CoreCompile:
正在跳过目标“CoreCompile”,因为所有输出文件相对于输入文件而言都是最新的。
StanalurJikecair:
123
GenerateBuildDependencyFile:
正在跳过目标“GenerateBuildDependencyFile”,因为所有输出文件相对于输入文件而言都是最新的。
GenerateBuildRuntimeConfigurationFiles:
正在跳过目标“GenerateBuildRuntimeConfigurationFiles”,因为所有输出文件相对于输入文件而言都是最新的。
CopyFilesToOutputDirectory:
RaskerYadeacorLalmi -> C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi\bin\Debug\netcoreapp2.0\RaskerYadeac
orLalmi.dll
已完成生成项目“C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi\RaskerYadeacorLalmi.csproj”(默认目标)的操作。
已成功生成。
0 个警告
0 个错误
上面代码创建的是 RaskerYadeacorLalmi
项目,在这个项目所在的文件夹进行编译,这个项目是新创建的,只是在项目上写了下面代码
<Target Name="StanalurJikecair" AfterTargets="CoreCompile">
<Message Text="123"></Message>
</Target>
这是一个空白的Target
,一个Target
有一个 Name 和一个属性告诉在什么时候运行这个Target
在这个Target
里使用Message
,就可以在编译时看到下面代码
StanalurJikecair:
123
如果重新运行msbuild
会发现有很多都是跳过,具体请看每次都要重新编译?太慢!让跨平台的 MSBuild/dotnet build 的 Target 支持差量编译 - walterlv
C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi>msbuild
用于 .NET Framework 的 Microsoft (R) 生成引擎版本 15.7.180.61344
版权所有(C) Microsoft Corporation。保留所有权利。
生成启动时间为 2018/8/3 20:05:47。
节点 1 上的项目“C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi\RaskerYadeacorLalmi.csproj”(默认目标)。
GenerateTargetFrameworkMonikerAttribute:
正在跳过目标“GenerateTargetFrameworkMonikerAttribute”,因为所有输出文件相对于输入文件而言都是最新的。
CoreGenerateAssemblyInfo:
正在跳过目标“CoreGenerateAssemblyInfo”,因为所有输出文件相对于输入文件而言都是最新的。
CoreCompile:
正在跳过目标“CoreCompile”,因为所有输出文件相对于输入文件而言都是最新的。
StanalurJikecair:
warning:123
GenerateBuildDependencyFile:
正在跳过目标“GenerateBuildDependencyFile”,因为所有输出文件相对于输入文件而言都是最新的。
GenerateBuildRuntimeConfigurationFiles:
正在跳过目标“GenerateBuildRuntimeConfigurationFiles”,因为所有输出文件相对于输入文件而言都是最新的。
CopyFilesToOutputDirectory:
RaskerYadeacorLalmi -> C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi\bin\Debug\netcoreapp2.0\RaskerYadeac
orLalmi.dll
已完成生成项目“C:\lindexi\github\RaskerYadeacorLalmi\RaskerYadeacorLalmi\RaskerYadeacorLalmi.csproj”(默认目标)的操作。
已成功生成。
0 个警告
0 个错误
如果需要清理,重新编译,可以输入下面命令
msbuild clean
现在可以尝试使用 Conditions
判断条件
使用 Conditions
很多时候都是使用字符串判断,如使用下面代码
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>netcoreapp2.0</TargetFramework>
</PropertyGroup>
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="$(TargetFramework)=='netcoreapp2.0'">
<Message Text="123"></Message>
</Target>
因为TargetFramework
是 netcoreapp2.0
会运行这个 Target ,如果这时修改 TargetFramework
为 net45
就不会运行这个代码
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="$(TargetFramework)=='net45'">
<Message Text="123"></Message>
</Target>
如果需要同时判断多个条件,如在 Debug
而且是 net45
就运行,可以使用下面代码
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="'$(Configuration)|$(TargetFramework)'=='DEBUG|net45'">
<Message Text="123"></Message>
</Target>
注意在原来的$(Configuration)
是可以不使用引号,但是现在使用了|
就需要添加引号,表示这时字符串
实际的|
不是语法,可以换为其他的字符,如下面的代码
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="'$(TargetFramework)-$(OutputType)'=='netcoreapp2.0-Exe'">
<Message Text="123"></Message>
</Target>
上面代码使用 '$(TargetFramework)-$(OutputType)'
判断
判断不相等
如果需要判断不相同,只需要修改==
为不相等
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="'$(TargetFramework)'!='net45'">
<Message Text="123"></Message>
</Target>
现在运行msbuild
可以看到输出了123
如果修改为'$(TargetFramework)'=='net45'
就判断不相等
判断大小
除了判断字符串,还可以判断字符串的大小,只能用来判断数值字符串,如果对于 16 进制的字符串,需要使用 0x
开始,如下面代码
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="'60'>'10'">
<Message Text="因为60大于10所以会运行"></Message>
</Target>
<Target Name="LamaciswhaJoopisJerall" AfterTargets="CoreCompile" Condition="'60'>='60'">
<Message Text="因为60大于60所以会运行"></Message>
</Target>
<Target Name="TasyenatarReresetelRagearsu" AfterTargets="CoreCompile" Condition="'10' < '60'">
<Message Text="因为10小于60所以会运行"></Message>
</Target>
<Target Name="FurkeeneLafer" AfterTargets="CoreCompile" Condition="'60'<='60'">
<Message Text="因为60小于等于60所以会运行"></Message>
</Target>
<Target Name="JoudewalljeeZeargeaday" AfterTargets="CoreCompile" Condition="'0xAA'>'10'">
<Message Text="如果使用16进制需要使用0x放在字符串最前"></Message>
</Target>
因为使用的文件是 xml 所以需要将会>
转换为>
,将<
转为 >
如果输入的数值是16进制就需要使用 0x 放在开始,下面的代码就会出现下面的错误
<Target Name="SiscooLecem" AfterTargets="CoreCompile" Condition="'AA'>'10'">
<Message Text="如果使用16进制需要使用0x放在字符串最前"></Message>
</Target>
error MSB4086: 尝试在条件“'AA
'>'10'”中对计算结果为“AA”而不是数字的“AA”进行数值比较
判断文件存在
在条件判可以用 Exists
判断文件 文件夹是否存在
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="Exists('$(OutputPath)')">
<Message Text="$(OutputPath)"></Message>
</Target>
<Target Name="ZurwelSowselnu" AfterTargets="CoreCompile" Condition="!Exists('$(OutputPath)')">
<Message Text="$(OutputPath)"></Message>
</Target>
可以看到两个代码的不相同,使用 !
可以判断为 原来是相同的就返回false
,这里的 $(OutputPath)
是存在的,所以编译会输出下面代码
StanalurJikecair:
bin\Debug\netcoreapp2.0\
判断多个条件
除了使用开始的使用 -
等连接多个判断还可以使用 And
Or
来判断多个条件,如下面代码
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="Exists('$(OutputPath)') And $(Configuration)=='Debug'">
<Message Text="$(OutputPath)"></Message>
</Target>
同时判断存在输出的文件夹并且在测试下才运行
注意不能使用引号加上 And 如'And'
,这时 And
会作为字符串
如果使用多个条件,建议使用()
包括多个条件,如下面代码,同时进行多个判断
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>netcoreapp2.0</TargetFramework>
<Foo>123</Foo>
</PropertyGroup>
<Target Name="StanalurJikecair" AfterTargets="CoreCompile" Condition="(Exists('$(OutputPath)') And $(Configuration)=='Debug') or ($(Foo)=='123')">
<Message Text="$(OutputPath)"></Message>
</Target>
使用的范围
在很多地方都可以使用条件进行判断,如放在任意的PropertyGroup
里,如果判断为 false 就不会定义这个属性
<PropertyGroup>
<OutputType Condition="'德熙' == '逗比'">Exe</OutputType>
<TargetFramework>netcoreapp2.0</TargetFramework>
</PropertyGroup>
也可以直接放在PropertyGroup
让整个PropertyGroup
不定义
<PropertyGroup Condition="'德熙' == '逗比'">
<OutputType>Exe</OutputType>
<TargetFramework>netcoreapp2.0</TargetFramework>
</PropertyGroup>
同样可以放在 ItemGroup
里,也可以放在 ItemGroup
控制是否定义
<ItemGroup>
<Foo Condition="'德熙' == '逗比'" Include="123"></Foo>
</ItemGroup>
<ItemGroup Condition="'德熙' == '逗比'">
<Foo Include="123"></Foo>
</ItemGroup>
还有上面写的 Target
也可以使用,在 Target
里也可以使用条件
<Target Name="StanalurJikecair" AfterTargets="CoreCompile">
<Message Condition="'德熙' == '逗比'" Text="$(OutputPath)"></Message>
</Target>
还可以写在 Import
一般在 Import
都需要先判断是否存在文件
<Import Condition="Exists('../lindexi.txt')" Project="../lindexi.txt"></Import>
特别感谢 HaibaraAi 大佬的阅读
本文会经常更新,请阅读原文: https://dotnet-campus.github.io//post/Roslyn-%E5%9C%A8%E9%A1%B9%E7%9B%AE%E6%96%87%E4%BB%B6%E4%BD%BF%E7%94%A8%E6%9D%A1%E4%BB%B6%E5%88%A4%E6%96%AD.html ,以避免陈旧错误知识的误导,同时有更好的阅读体验。
本作品采用 知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议 进行许可。欢迎转载、使用、重新发布,但务必保留文章署名 lindexi (包含链接: https://dotnet-campus.github.io/ ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请 与我联系 。