背景
很多情況下,我們編寫了一些工具庫之后,往往在某些框架版本中會出現一些問題,比如本人最近寫的一個導入導出的工具庫Magicodes.IE(GitHub:https://github.com/xin-lai/Magicodes.IE)就出現了以下問題:

雖然支持標准庫2.0,但是通過編寫單元測試,確實發現如此:

因此,單元測試確實需要針對各個框架版本進行測試,並且相關代碼需要針對框架進行兼容修改。那么如何實現呢?接下來我們一步步實踐。
.NET框架版本說明
最新目標框架版本
下表定義了最常見的目標框架、如何引用這些框架,以及它們實現的 .NET Standard 版本。 這些目標框架版本是最新的穩定版本。 預覽版不會顯示。 目標框架名字對象 (TFM) 是一個標准化令牌格式,用於指定 .NET 應用或庫的目標框架。
目標 Framework | 最新 穩定版本 | 目標框架名字對象 (TFM) | 已實現 .NET Standard 版本 |
---|---|---|---|
.NET Standard | 2.1 | netstandard2.1 | 不可用 |
.NET Core | 3.0 | netcoreapp3.0 | 2.1 |
.NET Framework | 4.8 | net48 | 2.0 |
支持的目標框架版本
目標框架通常由 TFM 引用。 下表顯示 .NET Core SDK 和 NuGet 客戶端支持的目標框架。 等效項顯示在括號內。 例如,win81
對於 netcore451
來說等效於 TFM。
目標 Framework | TFM |
---|---|
.NET Standard | netstandard1.0 netstandard1.1 netstandard1.2 netstandard1.3 netstandard1.4 netstandard1.5 netstandard1.6 netstandard2.0 netstandard2.1 |
.NET Core | netcoreapp1.0 netcoreapp1.1 netcoreapp2.0 netcoreapp2.1 netcoreapp2.2 netcoreapp3.0 |
.NET Framework | net11 net20 net35 net40 net403 net45 net451 net452 net46 net461 net462 net47 net471 net472 net48 |
Windows 應用商店 | netcore [netcore45] netcore45 [win] [win8] netcore451 [win81] |
.NET Micro Framework | netmf |
Silverlight | sl4 sl5 |
Windows Phone | wp [wp7] wp7 wp75 wp8 wp81 wpa81 |
通用 Windows 平台 | uap [uap10.0] uap10.0 [win10] [netcore50] |
如何讓工程支持多個框架?
以單元測試工程為例:
<PropertyGroup> <TargetFrameworks>netcoreapp3.0;netcoreapp2.2;netcoreapp3.1;net461</TargetFrameworks> <IsPackable>false</IsPackable> </PropertyGroup>
如上述代碼所示,我們可以通過“TargetFrameworks”元素來定義多個框架。那么如何在工程里面添加條件判斷以進行編譯呢?
<ItemGroup Condition=" '$(TargetFramework)' == 'net461' "> <PackageReference Include="Newtonsoft.Json" Version="12.0.3" /> </ItemGroup>
除了工程里面,我們代碼中如何針對不同的框架版本編寫代碼呢?
#if NET461 return excelPackage.Workbook.Worksheets[typeof(T).GetDisplayName()] ?? excelPackage.Workbook.Worksheets[ExcelImporterSettings.SheetName] ?? excelPackage.Workbook.Worksheets[1]; #else return excelPackage.Workbook.Worksheets[typeof(T).GetDisplayName()] ?? excelPackage.Workbook.Worksheets[ExcelImporterSettings.SheetName] ?? excelPackage.Workbook.Worksheets[0]; #endif
這些“NET461”的符號都有哪些呢?如下所示:
完整的 .NET Core 目標框架的預處理器符號列表
目標框架 | 符號 |
---|---|
.NET Framework | NETFRAMEWORK , NET20 , NET35 , NET40 , NET45 , NET451 , NET452 , NET46 , NET461 , NET462 , NET47 , NET471 , NET472 , NET48 |
.NET Standard | NETSTANDARD , NETSTANDARD1_0 , NETSTANDARD1_1 , NETSTANDARD1_2 , NETSTANDARD1_3 , NETSTANDARD1_4 , NETSTANDARD1_5 , NETSTANDARD1_6 , NETSTANDARD2_0 , NETSTANDARD2_1 |
.NET Core | NETCOREAPP , NETCOREAPP1_0 , NETCOREAPP1_1 , NETCOREAPP2_0 , NETCOREAPP2_1 , NETCOREAPP2_2 , NETCOREAPP3_0 , NETCOREAPP3_1 |
Magicodes.IE之多框架版本測試
了解了這些知識,我們就可以編寫多框架版本的實現和單元測試了。
單元測試添加多框架支持
-
修改TargetFramework為TargetFrameworks
-
設置分組
根據目標框架分組:
-
修復編譯錯誤
-
執行單元測試
單元測試結果如下圖所示:
針對具體的單元測試,我們還可以指定目標框架進行運行和調試:
總結和經驗分享
結合整改的過程,我們可以得出以下經驗:
- 單元測試編寫時的異常消息判斷盡量使用字符串包含判斷,而不判斷完整格式。

如上面所示,左側代碼在net461、netcoreapp2.2都是無法通過的,格式化的參數模板不一致。
-
特定框架、平台代碼需添加符號判斷
如下面示例,默認情況下,.NET Core 不提供除代碼頁 28591 以外的其他任何代碼頁編碼和 Unicode 編碼,例如 UTF-8 和 UTF-16,所以我們可以使用以下代碼進行添加,但是需要排除.NET Framework:
除了特定代碼之外,有時還有特定依賴的包:
-
慎用新語法糖
比如以下代碼:
精力有限,暫不適配過多框架版本,有興趣的朋友可以參與進來。