作为 .NET 系列的最新成员,.NET Core 和 .NET Standard 的概念及其与 .NET Framework 的区别并不十分明确。在本文中,我将准确介绍每个产品及其适用场景。

在详细介绍之前,建议先审视一下 .NET 的总体情况,了解 .NET Core 和 .NET Standard 是如何在这一体系中发挥作用的。.NET Framework 是在 15 年前首次推出(我怎么记得2008年时我就在用.net framework1.1了?),当时只有一个 .NET 堆栈可用于生成 Windows 桌面和 Web 应用程序。从那以后,出现了其他 .NET 实现。例如,可用于生成 iOS/Android 移动应用和 macOS 桌面应用程序的 Xamarin,如图 1 所示。

:.NET 总体情况

下面说明了 .NET Core 和 .NET Standard 是如何在这一体系中发挥作用的:

  • .NET Core:这是最新的 .NET 实现。它不仅开放源代码,而且还适用于多个 OS。使用 .NET Core,可以生成跨平台的控制台应用程序、ASP.NET Core Web 应用程序和云服务。
  • .NET Standard:这是所有 .NET 实现都必须实现的一组基本 API,通常称为"基类库 (BCL)"。通过定目标到 .NET Standard,可以生成能跨所有 .NET 应用程序共享的库,无论它们是在哪个 .NET 实现或 OS 上运行。

.NET Core 简介

作为 .NET Framework 和 Silverlight 分支,.NET Core 是全新的 .NET 实现,不仅跨平台,而且还完全开放源代码。通过启用独立式 XCOPY 部署,它已经过优化,更适用于移动和服务器工作负载。

为了能够更好地认识 .NET Core,接下来将深入了解如何进行 .NET Core 开发。为此,将同时探索新的命令行工具。也可以使用 Visual Studio 2017 进行 .NET Core 开发,但本文读者很可能已相当熟悉 Visual Studio,因此我将重点介绍新体验。

在 .NET 创建之初,它经过了大量优化,以便用户能够在 Windows 上快速开发应用程序。实际上,这意味着 .NET 开发与 Visual Studio 是不可分开的。可以肯定的是:使用 Visual Studio 进行开发是一种奇妙的感受。这样开发的工作效率超高,调试程序是我使用过的最好用的。

不过,有时,使用 Visual Studio 并不是最便捷的选择。假设只是想通过 .NET 学习 C#。在这种情况下,就无需下载并安装好几个 GB 的 IDE。或者,假设要通过 SSH 访问 Linux 计算机。在这种情况下,使用 IDE 根本就行不通。又或者,假设只是喜欢使用命令行接口 (CLI)。

正因为此,名为 .NET Core CLI 的一流 CLI 诞生了。.NET Core CLI 的主驱动程序称为"dotnet"。 它可用于开发的几乎所有方面,包括创建、生成、测试和打包项目。接下来,将了解具体操作。

首先,创建并运行 Hello World 控制台应用程序(我使用的是 Windows PowerShell,而在 macOS 或 Linux 上使用 Bash 也同样有效):(命令行下创建项目、编译,对我这样的老年选手来除了装逼应该没什么别的用处了,当然我也没心思记住它们)

$ dotnet new console -o hello

$ cd hello

$ dotnet run

Hello World!

CLI 中的"dotnet new"命令相当于 Visual Studio 中的"文件 | 新建项目"。可以创建各种不同类型的项目。键入"dotnet new"可以查看预安装的各种模板。

现在,将一些逻辑提取到类库中。为此,请先创建与"hello"项目平行的类库项目:

$ cd ..

$ dotnet new library -o logic

$ cd logic

由于要封装的逻辑是构造 Hello World 消息,因此将 Class1.cs 的内容更改为以下代码:

namespace logic

{

publicstaticclass HelloWorld

{

publicstaticstring GetMessage(string name) => $"Hello {name}!";

}

}

此时,还应将 Class1.cs 重命名为 HelloWorld.cs:

$ mv Class1.cs HelloWorld.cs

请注意,无需因为有此更改而更新项目文件。.NET Core 中使用的新项目文件只包括项目目录中的所有源文件。因此,添加、删除和重命名文件不再需要修改项目。这样一来,可以更顺畅地使用命令行执行文件操作。

若要使用 Hello World 类,需要将 hello 应用程序更新为引用逻辑库。为此,可以编辑项目文件,也可以使用 dotnet add reference 命令:

$ cd ../hello

$ dotnet add reference ../logic/logic.csproj

现在,将 Program.cs 文件更新为使用 HelloWorld 类,如图 2 所示。

图 2:将 Program.cs 文件更新为使用 HelloWorld 类

using System;

using logic;

namespace hello

{

class Program

{

staticvoid Main(string[] args)

{

Console.Write("What's your name: ");

var name = Console.ReadLine();

var message = HelloWorld.GetMessage(name);

Console.WriteLine(message);

}

}

}

若要生成并运行应用程序,只需键入 dotnet run:

$ dotnet run

What's your name: Immo

Hello Immo!

还可以通过命令行创建测试。CLI 支持 MSTest 和常用的 xUnit 框架。此示例使用 xUnit:

$ cd ..

$ dotnet new xunit -o tests

$ cd tests

$ dotnet add reference ../logic/logic.csproj

更改 UnitTest1.cs 内容以添加测试,如图 3 所示。

图 3:更改 UnitTest1.cs 内容以添加测试

using System;

using Xunit;

using logic;

namespace tests

{

publicclass UnitTest1

{

[Fact]

publicvoid Test1()

{

var expectedMessage = "Hello Immo!";

var actualMessage = HelloWorld.GetMessage("Immo");

Assert.Equal(expectedMessage, actualMessage);

}

}

}

现在,可以调用 dotnet test 运行测试:

$ dotnet test

Total tests: 1. Passed: 1. Failed: 0. Skipped: 0.

Test Run Successful.

为了让操作变得更有趣一点,将创建简单的 ASP.NET Core 网站:

$ cd ..

$ dotnet new web -o web

$ cd web

$ dotnet add reference ../logic/logic.csproj

编辑 Startup.cs 文件,并将 app.Run 调用更改为使用 HelloWorld 类,如下所示:

app.Run(async (context) =>

{

var name = Environment.UserName;

var message = logic.HelloWorld.GetMessage(name);

await context.Response.WriteAsync(message);

});

若要启动开发 Web 服务器,只需再次运行 dotnet run:

$ dotnet run

Hosting environment: Production

Now listening on: http://localhost:5000

Application started. Press Ctrl+C to shut down.

转到所显示的 URL(应为 http://localhost:5000)。

此时,项目结构应如图 4 所示。

图 4:创建的项目结构

$ tree /f

├───hello

│ hello.csproj

│ Program.cs

├───logic

│ HelloWorld.cs

│ logic.csproj

├───tests

│ tests.csproj

│ UnitTest1.cs

└───web

Program.cs

Startup.cs

web.csproj

为了能够更方便地使用 Visual Studio 编辑文件,还将创建解决方案文件,并将所有项目添加到解决方案中:

$ cd ..

$ dotnet new sln -n HelloWorld

$ ls -fi *.csproj -rec | % { dotnet sln add $_.FullName }

可以看到,.NET Core CLI 功能非常强大,能够为具有其他背景的开发者提供相当熟悉的精益体验。虽然此过程是将 dotnet 与 Windows PowerShell 结合使用,但如果是在 Linux 或 macOS 上,使用体验也会非常相似。

.NET Core 的另一巨大优势在于,它支持独立式部署。可以使用 Docker 对应用程序进行容器化处理,以便它能够有自己的 .NET Core 运行时副本。这样一来,可以在使用不同版本 .NET Core 的同一台计算机上运行各种应用程序,而它们则互不干扰。由于 .NET Core 开放源代码,因此还可以添加每日版或甚至自己修改或生成的版本,其中可能包括自己所做的修改。不过,这并不在本文的讨论范围之内。

.NET Standard 简介

在开发者生成新式体验时,应用程序通常跨越多种外形规格,因而也就跨越多个 .NET 实现。今时今日,客户非常希望可以在手机上使用 Web 应用程序,并希望能够通过基于云的后端共享数据。使用笔记本电脑时,他们也希望可以通过网站获得访问权限。对于自己的基础结构,很可能希望可以借助命令行工具和潜在的桌面应用程序,允许员工管理系统。请参阅图 5,了解具有此类用途的不同 .NET 实现。

:介绍了 .NET 实现

OS

开源

用途

.NET Framework

Windows

N

用于生成在 IIS 上运行的 Windows 桌面应用程序和 ASP.NET Web 应用程序。

.NET Core

Windows、Linux、macOS

Y

用于生成跨平台的控制台应用程序、ASP.NET Core Web 应用程序和云服务。

Xamarin

iOS、Android、macOS

Y

用于生成适用于 iOS 和 Android 的移动应用,以及适用于 macOS 的桌面应用程序。

.NET Standard

Y

用于生成可以从所有 .NET 实现(如 .NET Framework、.NET Core 和 Xamarin)引用的库。

在这样的环境下,代码共享成为重大挑战。需要了解 API 的可用位置,并确保共享组件仅使用跨所有要用 .NET 实现都支持的 API。

此时,.NET Standard 就派上用场了。.Net Standard 是一种规范。每个 .NET Standard 版本都定义了一组 API。为了与相应版本保持一致,所有 .NET 实现都必须提供这些 API。可以将 .NET Standard 看作是另一个 .NET 堆栈,不同之处在于无法为其生成应用程序,只能生成库。若希望可以从任意位置引用库,应对库使用此 .NET 实现。

大家不禁想知道,.NET Standard 涵盖了哪些 API。如果熟悉 .NET Framework,就应该熟悉我之前提到的 BCL。BCL 是与 UI 框架和应用程序模型无关的一组基础 API。它包括基元类型、文件 I/O、网络、反射、序列化、XML 等。

所有 .NET 堆栈都会实现某版 .NET Standard。经验法则是,生成新版 .NET 实现时,通常都会实现最新版 .NET Standard。

很贴切的例子是 HTML 和浏览器:将 HTML 规范看作是 .NET Standard,将不同的浏览器看作是 .NET 实现,如 .NET Framework、.NET Core 和 Xamarin。

此时,大家可能非常好奇,如何使用 .NET Standard。实际上,大家已经知道了。还记得之前创建逻辑类库时的情形吗? 接下来,将深入了解一下项目文件:

$ cd logic

$ cat logic.csproj

<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>

<TargetFramework>netstandard2.0</TargetFramework>

</PropertyGroup>

</Project>

将此文件与"hello"控制台应用程序项目文件进行对比:

$ cd ..\hello

$ cat hello.csproj

<Project Sdk="Microsoft.NET.Sdk">

<ItemGroup>

<ProjectReference Include="..\logic\logic.csproj" />

</ItemGroup>

<PropertyGroup>

<OutputType>Exe</OutputType>

<TargetFramework>netcoreapp2.0</TargetFramework>

</PropertyGroup>

</Project>

可以看到,逻辑库的 TargetFramework 值为 netstandard2.0,而控制台应用程序的值为 netcoreapp2.0。TargetFramework 属性指明要定目标到的 .NET 实现。因此,控制台应用程序定目标到 .NET Core 2.0,而库定目标到 .NET Standard 2.0。也就是说,不仅可以从 .NET Core 应用程序引用逻辑库,还可以从生成的 .NET Framework 或 Xamarin 应用程序引用逻辑库。

遗憾的是,目前可用的大多数库都尚未定目标到 .NET Standard。大多数库都定目标到 .NET Framework。当然,并不是所有库都可以(或甚至应该)定目标到 .NET Standard。例如,包含 Windows Presentation Foundation (WPF) 控件的库需要定目标到 .NET Framework,因为 UI 并不是标准的一部分。不过,许多常规用途库定目标到 .NET Framework 只是因为,在它们创建时还没有 .NET Standard。

在 .NET Standard 2.0 推出后,API 集变得非常大,大多数(如果不是全部的话)常规用途库可以定目标到 .NET Standard。因此,目前 NuGet 上的全部库中有 70% 只使用现在属于 .NET Standard 的 API。尽管如此,其中只有一小部分明确标注为与 .NET Standard 兼容。

为了让开发者无障碍地使用它们,添加了一种兼容性模式。如果安装的 NuGet 包没有为目标框架提供库,也没有为 .NET Standard 提供库,那么 NuGet 会转而求助于 .NET Framework。也就是说,可以引用 .NET Framework 库,就像是定目标到 .NET Standard 一样。

我将演示一下具体操作。在此示例中,我将使用名为 PowerCollections 的常用集合库,它是在 2007 年编写的。它有一段时间没有更新了,仍定目标到 .NET Framework 2.0。我将从 NuGet 将此库安装到 hello 应用程序中:

$ dotnet add package Huitian.PowerCollections

此库提供了 BCL 没有的其他集合类型,如不保证排序的包。接下来,将把 hello 应用程序更改为使用此库,如图 6 所示。

图 6:使用 PowerCollections 的示例应用程序

using System;

using Wintellect.PowerCollections;

namespace hello

{

class Program

{

staticvoid Main(string[] args)

{

var data = new Bag<int>() { 1, 2, 3 };

foreach (var element in data)

Console.WriteLine(element);

}

}

}

如果运行程序,将会看到以下警告:

$ dotnet run

hello.csproj : warning NU1701: Package 'Huitian.PowerCollections 1.0.0' was restored using'.NETFramework,Version=v4.6.1' instead of the project target framework '.NETCoreApp,Version=v2.0'. This may cause compatibility problems.

1

3

2

那么,究竟发生了什么? hello 应用程序定目标到 .NET Core 2.0。由于 .NET Core 2.0 实现的是 .NET Standard 2.0,因此还有可引用 .NET Framework 库的兼容性模式。不过,并不是所有 .NET Framework 库都适用于各种 .NET 实现。例如,它们可能会使用 Windows 窗体或 WPF API。由于无法确定具体情况,因此 NuGet 会显示警告消息,提醒注意这种情况,这样就不会浪费时间排查可能由此导致的问题了。

请注意,每次生成时都会看到此警告。这样可避免在包安装期间根本看不到警告或忘记这种情况的问题。

当然,没有什么比每次生成时都需要忽略无法作为行动依据的警告更糟糕的了。因此,建议在验证应用程序后,为相应包禁用警告。由于应用程序运行正常(正确输出了所创建的包的内容),因此现在可以取消警告。为此,请编辑 hello.csproj 文件,并将 NoWarn 属性添加到包引用中:

<PackageReference Include="Huitian.PowerCollections" Version="1.0.0"

NoWarn="NU1701" />

如果现在重新运行应用程序,就再也不会看到警告了。如果安装另一个使用兼容性模式的包,也会看到可以取消的相应包警告。

使用新工具,还可以让类库项目创建属于生成的 NuGet 包。这样一来,可以更轻松地与全世界共享库(通过推送到 nuget.org),也可以仅在组织内共享(通过推送到大家在 Visual Studio Team Services 或 MyGet 上自己的包源)。新项目还支持多重定目标,这样就可以生成一个适用于多个 .NET 实现的项目。也就是说,可以使用条件编译 (#if) 来调整库,使其适应特定的 .NET 实现。还可以生成平台专属 API 的 .NET Standard 包装器。不过,这些都不在本文的讨论范围之内。

结束语

.NET Standard 是所有 .NET 实现都必须提供的 API 规范。这样一来,可以让 .NET 系列保持一致,并生成能够从任何 .NET 实现引用的库。它取代了生成共享组件的 PCL。

.NET Core 是更适合使用 ASP.NET Core 生成控制台应用程序、Web 应用程序和云服务的 .NET Standard 实现。它的 SDK 随附功能非常强大的工具,除了支持 Visual Studio 开发外,还支持基于命令行的完整开发工作流。

.NET Standard - 揭秘 .NET Core 和 .NET Standard[转自MSDN]的更多相关文章

  1. 错误处理:java.lang.NoClassDefFoundError: org/apache/taglibs/standard/tag/rt/core/ForEachTag

    在使用JSP.Servlet进行开发时,遇到java.lang.NoClassDefFoundError: org/apache/taglibs/standard/tag/rt/core/ForEac ...

  2. .NET Core和.NET Standard有什么不同

        近日,微软发布了.NET Core 2.0,但是开发人员中间仍然存在一些疑惑,就是.NET Core..NET Standard.Xamarin和.NET Framework有什么不同. .N ...

  3. .NET Standard vs. .NET Core

    What is the difference between .NET Core and .NET Standard Class Library project types? Answer1 When ...

  4. .NET Core和.NET Standard

    作为.NET家族的最新成员,有很多关于.NET Core和.NET Standard的误解,以及它们于.NET Framework之间的区别.在这篇文章,我会准确的解释他们究竟是什么,并看看何时应选择 ...

  5. .NET Core、.NET Standard、Xamarin和.NET Framework对比

    近日,微软发布了.NET Core 2.0,但是开发人员中间仍然存在一些疑惑,就是.NET Core..NET Standard.Xamarin和.NET Framework有什么不同. .NET F ...

  6. .Net Standard(.Net Core)实现获取配置信息

    一.前言 在.Net Framework框架有专门获取webconfig配置的方法供我们使用,但是在.Net Core或者.Net Standard中没有可以直接使用的方法来获取配置文件信息,下面就来 ...

  7. .NET Standard和.NET Core是什么关系(转载)

    .NET Standard vs .NET Core 问: I have read about the difference between .NET Standard and .NET Core, ...

  8. 你不得不知道的 .NET CORE —— .NET Framework, .NET Core 和 .NET Standard 的区别

    .NET Framework 和 .NET Core 是平台应用框架,而 .NET Standard 是 .NET 底层库.因此只要用 .NET Standard 工程来写的代码可以直接在上层的平台应 ...

  9. .Net Core和.Net Standard直观理解

    .NET framework和.NET Core里面有一些部分,内容是相同的. 这部分相同的内容,就被称为标准库...即NET Standard Library. 而那些不同的部分,则分别叫做.NET ...

随机推荐

  1. jQuery序列化表单数据 serialize()、serializeArray()及使用

    1.serialize() 方法: serialize() 方法通过序列化表单值,创建 URL 编码文本字符串. 您可以选择一个或多个表单元素(比如 input 及/或 文本框),或者 form 元素 ...

  2. MapReduce中的分布式缓存使用

    MapReduce中的分布式缓存使用 @(Hadoop) 简介 DistributedCache是Hadoop为MapReduce框架提供的一种分布式缓存机制,它会将需要缓存的文件分发到各个执行任务的 ...

  3. 恢复计算机崩溃数据的五款最佳Linux发行版

    嗨,Linux 新手们!你们在尝试运行命令时有没有搞坏过计算机系统?我相信你们有过这种经历.这一幕经常发生:你想尝试运行命令,或者安装测试更新版,结果下一次重启时计算机就崩溃了.我在本文将逐一介绍五款 ...

  4. (剑指Offer)面试题52:构建乘积数组

    题目: 给定一个数组A[0,1,...,n-1],请构建一个数组B[0,1,...,n-1],其中B中的元素B[i]=A[0]*A[1]*...*A[i-1]*A[i+1]*...*A[n-1].不能 ...

  5. 关于substring的char[]共享

    我们知道,对于一个较大的String对象假设从中获取一个子串.jdk默认子串的char[]是共享原串的char[].即子串的char[]是原串的char[]中的一部分, 这样对于一个原串多个子串的情况 ...

  6. Ubuntu下压缩包内文件解压后乱码问题的解决

    用到的工具是The Unarchiver项目提供的lsar/unar工具. The Unarchiver项目主页:http://code.google.com/p/theunarchiver/ 安装( ...

  7. oracle中database links的使用

    1.在pl/sql developer中创建database links 2.使用database links select * from table1@xtbg 注意:@xtbg是database ...

  8. Java并发性和多线程介绍目录

    http://ifeve.com/java-concurrency-thread-directory/

  9. 说说PHP中的命名空间相关概念

    说说PHP中的命名空间相关概念 1. PHP中的命名空间是什么? 什么是命名空间?"从广义上来说,命名空间是一种封装事物的方法.在非常多地方都能够见到这样的抽象概念. 比如.在操作系统中文件 ...

  10. 算法笔记_037:寻找和为定值的两个数(Java)

    目录 1 问题描述 2 解决方案 2.1 排序夹逼法   1 问题描述 输入一个整数数组和一个整数,在数组中查找两个数,满足他们的和正好是输入的那个整数.如果有多对数的和等于输入的整数,输出任意一对即 ...