在引用多目标类库时,如何让 NuGet 知道包的特定版本?

问题描述

我们有三个项目:

WebApp.csproj,针对 .NET Framework 2.0 编译的 ASP.NET WebForms 应用

Api.csproj,针对 .NET Framework 4.5 编译的 ASP.NET WebApi 2

Lib.csproj,针对 .NET Framework 2.0 编译并被上述两个项目引用。

Lib.csproj 已以新的 .NET 项目格式(.NET Core 使用的格式)重新创建,并使用 <TargetFrameworks>net20;net45</TargetFrameworks> 技术针对 .NET Fw 2.0 和 4.5。

到目前为止一切顺利,我可以通过在 Lib.csproj 中编写适当的部分来引用不同的 NuGet 包,如下所示:

<ItemGroup Condition="'$(TargetFramework)' == 'net20'">    
    <packagereference Include="NLog" Version="2.1.0.0" />
</ItemGroup>

<ItemGroup Condition="'$(TargetFramework)' == 'net45'">
    <packagereference Include="NLog" Version="4.7.8.0" />
</ItemGroup>

并且我可以将 Api.csproj 指向以 Lib.csproj 为目标的 net45 版本,如下所示:

<ProjectReference Include="..\..\Lib\Lib.csproj" AdditionalProperties="TargetFramework=net45">
    <Project>{32e77bcf-152c-4b64-be37-a13f49cdcab6}</Project>
    <Name>Lib</Name>
</ProjectReference>

net20 版本引用 NLog 2.1.0net45 一个 NLog 4.7.8

问题:现在我想在 NLog 4.7.8 中使用 Api.csproj,但是当我在包含 Manage Packages for SolutionApi.csproj解决方案上单击 Lib.csproj 时,引用的版本似乎是 2.1.0

我尝试选择两个项目并安装它,但是我正确地收到错误,因为 NLog 4.7.8net20 不兼容。

实际上,问题在于 NuGet 似乎没有意识到我引用的项目是多目标项目,并且只能看到安装在 net20 版本中的 NLog 版本,即使 { {1}} 的目标是 Api.csproj 的正确“风味”(如果我去检查二进制文件,复制到输出文件夹的 NLog.dll 版本对于我的每个目标都是正确的) .

我尝试在互联网上查找,但无济于事。

公平地说:复制到 Lib.csprojApi.csproj 文件夹的 NLog 版本是正确的,所以它不是一个表演者。但这有点烦人,因为 NuGet 显示了我的依赖项的错误版本,然后很难判断我们实际使用的是哪个版本。

对于一些上下文:这个应用程序显然是古老的,至少 bin。由于性能原因,我们让 WebApp.csproj 使用 Api.csproj,并且由于该项目引用了可以在 async/await 中异步但显然不会在 .NET 2 下编译的代码,我们选择使 Lib.csproj 成为多目标。

解决方法

此处的 AdditionalProperties 不是必需的:

<ProjectReference Include="..\..\Lib\Lib.csproj" AdditionalProperties="TargetFramework=net45">
    <Project>{32e77bcf-152c-4b64-be37-a13f49cdcab6}</Project>
    <Name>Lib</Name>
</ProjectReference>

如果可能,我强烈建议您更新所有项目以使用 SDK 样式的 csproj 格式。这对于遗留(核心前)网络项目是不可能的,WebApp.csprojApi.csproj 就是这种情况。但是,即使是这些项目也可以使用某些部分较新的 csproj 格式。

具体来说,they can use PackageReference。我假设您现有的 Web 项目正在使用 nuget.config,但可以将它们更改为使用 PackageReference,而无需实际使它们成为 SDK 样式的 csprojs。使用 nuget.config 时,NuGet 在传递依赖项方面存在一些问题,因此我只建议在任何地方使用 PackageReference