C# 发展历史

本页介绍了 C# 语言每个主要版本的发展历史。 C# 团队将继续创新,以添加新功能。 可以在 GitHub 上的 dotnet/roslyn 存储库上找到详细的语言功能状态,包括考虑在即将发布的版本中添加的功能。

重要

为了提供一些功能,C# 语言依赖 C# 规范定义为标准库 所用的类型和方法。 .NET 平台通过许多包交付这些类型和方法。 例如,异常处理。 为了确保引发的对象派生自 Exception,将会检查每个 throw 语句或表达式。 同样,还会检查每个 catch,以确保捕获的类型派生自 Exception。 每个版本都可能会新增要求。 若要在旧版环境中使用最新语言功能,可能需要安装特定库。 每个特定版本的页面中记录了这些依赖项。 若要了解此依赖项的背景信息,可以详细了解语言与库的关系

C# 1.0 版

回想起来,和 Visual Studio .NET 2002 一起发布的 C# 版本 1.0 非常像 Java。 在 ECMA 制定的设计目标中,它旨在成为一种“简单、现代、面向对象的常规用途语言”。 当时,它和 Java 类似,说明已经实现了上述早期设计目标。

不过如果现在回顾 C# 1.0,你会觉得有点晕。 它没有习以为常的内置异步功能和以泛型为中心的巧妙功能。 其实它完全不具备泛型。 那 LINQ 呢? 尚不可用。 这些新增内容需要几年才能推出。

与现在的 C# 相比,C# 1.0 版少了很多功能。 你会发现自己的代码很冗长。 不过凡事总要有个开始。 在 Windows 平台上,C# 1.0 版是 Java 的一个可行的替代之选。

C# 1.0 的主要功能包括:

C# 版本 1.2

随 Visual Studio .NET 2003 一起提供的 C# 版本 1.2。 它对语言做了一些小改进。 最值得注意的是,从此版本开始,当 IEnumerator 实现 IDisposable 时,foreach 循环中生成的代码会在 IEnumerator 上调用 Dispose

C# 2.0 版

从此以后事情变得有趣起来。 让我们看看 C# 2.0(2005 年发布)和 Visual Studio 2005 中的一些主要功能:

除现有功能以外的其他 C# 2.0 功能:

  • getter/setter 单独可访问性
  • 方法组转换(委托)
  • 静态类
  • 委托推断

C# 一开始是面向对象的 (OO) 通用语言,而 C# 2.0 版很快改变了这一点。 做好基础准备后,他们开始追求解决一些严重影响开发者的难点。 且他们以显著的方式追求这些难点。

通过泛型,类型和方法可以操作任意类型,同时保持类型安全性。 例如,通过 List<T>,将获得 List<string>List<int> 并且可以对这些字符串或整数执行类型安全操作,同时对其进行循环访问。 使用泛型优于创建派生自 ArrayListListInt 类型,也优于从每个操作的 Object 强制转换。

C# 2.0 版引入了迭代器。 简单来说,迭代器允许使用 foreach 循环来检查 List(或其他可枚举类型)中的所有项。 拥有迭代器是该语言最重要的一部分,显著提升了语言的可读性以及人们推出代码的能力。

不过 C# 依然在追赶 Java 的道路上。 当时 Java 已发布包含泛型和迭代器的版本。 但是随着语言各自的演化,形势很快发生了变化。

C# 3.0 版

C# 3.0 版和 Visual Studio 2008 一起发布于 2007 年下半年,但完整的语言功能是在 .NET Framework 3.5 版中发布的。 此版本标示着 C# 发展过程中的重大更改。 C# 成为了真正强大的编程语言。 我们来看看此版本中的一些主要功能:

回顾过去,这些功能中大多数似乎都是不可或缺,难以分割的。 它们的组合都是经过巧妙布局。 我们通常认为 C# 版本的杀手锏是查询表达式,也就是语言集成查询 (LINQ)。

LINQ 的构造可以建立在更细微的视图检查表达式树、Lambda 表达式以及匿名类型的基础上。 不过无论如何 C# 3.0 都提出了革命性的概念。 C# 3.0 开始为 C# 转变为面向对象/函数式混合语言打下基础。

具体来说,你现在可以编写 SQL 样式的声明性查询对集合以及其他项目执行操作。 无需再编写 for 循环来计算整数列表的平均值,现在可改用简单的 list.Average() 方法。 组合使用查询表达式和扩展方法让各种数字变得智能多了。

人们需要一些时间来掌握和吸收这种概念,不过已经逐渐做到了。 现在又过了几年,代码变得更简洁,功能也更强大了。

C# 4.0 版

C# 版本 4.0 随 Visual Studio 2010 一起发布,很难达到版本 3.0 的创新水平。 在 3.0 版中,C# 已经完全从 Java 的阴影中脱颖而出,崭露头角。 很快成为一种简洁精炼的语言。

下一版本引入了一些有趣的新功能:

嵌入式互操作类型缓解了为应用程序创建 COM 互操作程序集的部署难题。 泛型协变和逆变提供了更强的功能来使用泛型,但风格比较偏学术,应该最受框架和库创建者的喜爱。 命名参数和可选参数帮助消除了很多方法重载,让使用更方便。 但是这些功能都没有完全改变模式。

主要功能是引入 dynamic 关键字。 在 C# 4.0 版中引入 dynamic 关键字让用户可以替代编译时类型上的编译器。 通过使用 dynamic 关键字,可以创建和动态类型语言(例如 JavaScript)类似的构造。 可以创建 dynamic x = "a string" 再向它添加六个,然后让运行时理清下一步操作。

动态绑定存在出错的可能性,不过同时也为你提供了强大的语言功能。

C# 5.0 版

C# 版本 5.0 随 Visual Studio 2012 一起发布,是该语言有针对性的一个版本。 对此版本中所做的几乎所有工作都归入另一个突破性语言概念:适用于异步编程的 asyncawait 模型。 下面是主要功能列表:

另请参阅

调用方信息特性让你可以轻松检索上下文的信息,不需要采用大量样本反射代码。 这在诊断和日志记录任务中也很有用。

但是 asyncawait 才是此版本真正的主角。 C# 在 2012 年推出这些功能时,将异步引入语言作为最重要的组成部分,另现状大为改观。 如果你以前处理过冗长的运行操作以及实现回调的 Web,应该会爱上这项语言功能。

C# 6.0 版

C# 在 3.0 版和 5.0 版对面向对象的语言添加了主要的新功能。 版本 6.0 随 Visual Studio 2015 一起发布,通过该版本,它不再推出主导性的杀手锏,而是发布了很多使得 C# 编程更有效率的小功能。 以下介绍了部分功能:

其他新功能包括:

  • 索引初始化表达式
  • Catch/Finally 块中的 Await
  • 仅限 getter 属性的默认值

这些功能每一个都很有趣。 但从整体来看,可以发现一个有趣的模式。 在此版本中,C# 消除语言样本,让代码更简洁且更具可读性。 所以对喜欢简洁代码的用户来说,此语言版本非常成功。

除了发布此版本,他们还做了另一件事,虽然这件事本身与传统的语言功能无关。 他们发布了 Roslyn 编译器即服务。 C# 编译器现在是用 C# 编写的,你可以使用编译器作为编程工作的一部分。

C# 7.0 版

C# 7.0 版已与 Visual Studio 2017 一起发布。 虽然该版本继承和发展了 C# 6.0,但不包含编译器即服务。 以下介绍了部分新增功能:

其他功能包括:

这些都为开发者提供了很棒的新功能,帮助编写比以往任何时候都简洁的代码。 重点是缩减了使用 out 关键字的变量声明,并通过元组实现了多个返回值。

但 C# 的用途更加广泛了。 .NET Core 现在面向所有操作系统,着眼于云和可移植性。 语言设计者除了推出新功能外,也会在这些新功能方面付出时间和精力。

C# 7.1 版

C# 已开始随 C# 7.1 发布单点发行 。 此版本增加了语言版本选择配置元素、三个新的语言功能和新的编译器行为。

此版本中新增的语言功能包括:

最后,编译器有 -refout-refonly 两个选项,可用于控制引用程序集生成

C# 7.2 版

C# 7.2 版添加了几个小型语言功能:

C# 7.3 版

C# 7.3 版本有两个主要主题。 第一个主题提供使安全代码的性能与不安全代码的性能一样好的功能。 第二个主题提供对现有功能的增量改进。 此外,在此版本中添加了新的编译器选项。

以下新增功能支持使安全代码获得更好的性能的主题:

对现有功能进行了以下增强:

  • 可以使用元组类型测试 ==!=
  • 可以在多个位置使用表达式变量。
  • 可以将属性附加到自动实现的属性的支持字段。
  • in 区分的参数的方法解析得到了改进。
  • 重载解析的多义情况现在变得更少。

新的编译器选项为:

  • -publicsign,用于启用程序集的开放源代码软件 (OSS) 签名。
  • -pathmap用于提供源目录的映射。

C# 8.0 版

C# 8.0 版是专门面向 .NET C# Core 的第一个主要 C# 版本。 一些功能依赖于新的 CLR 功能,而其他功能依赖于仅在 .NET Core 中添加的库类型。 C# 8.0 向 C# 语言添加了以下功能和增强功能:

默认接口成员需要 CLR 中的增强功能。 这些功能已添加到 .NET Core 3.0 的 CLR 中。 范围和索引以及异步流需要 .NET Core 3.0 库中的新类型。 在编译器中实现时,可为 null 的引用类型在批注库时更有用,因为它可以提供有关参数和返回值的 null 状态的语义信息。 这些批注将添加到 .NET Core 库中。

C# 9.0 版

C# 9.0 随 .NET 5 一起发布。 它是面向 .NET 5 版本的任何程序集的默认语言版本。 它包含以下新功能和增强功能:

C# 9.0 向 C# 语言添加了以下功能和增强功能:

C# 9.0 继续以前版本中的三大主题:删除不必要的模式、将数据与算法分离,以及在更多位置提供更多模式。

顶级语句意味着主程序将更易于读取。 减少了不必要的模式:命名空间、Program 类和 static void Main() 都是不必要的。

records 的引入为遵循值语义的引用类型提供了简洁的语法,以实现相等性。 你将使用这些类型来定义通常定义最小行为的数据容器。 仅限 Init 的资源库在记录中提供了非破坏性修改功能(with 表达式)。 C# 9.0 还添加了协变返回类型,以便派生记录可以重写虚拟方法,并返回从基方法的返回类型派生的类型。

模式匹配功能以多种方式进行了扩展。 数值类型现在支持 范围模式。 可以使用 andornot 模式组合模式。 可以通过添加括号来阐明更复杂的模式。

另一组功能支持 C# 中的高性能计算:

  • nintnuint 类型对目标 CPU 上的本机大小整数类型进行建模。
  • 函数指针提供类似委托的功能,同时避免创建委托对象所需的分配。
  • localsinit 指令可以省略以保存指令。

另一组改进支持代码生成器添加功能的场景:

  • 模块初始化表达式是程序集加载时运行时调用的方法。
  • 分部方法支持新的可访问修饰符和非 void 返回类型。 在这些情况下,必须提供一个实现。

C# 9.0 添加了许多其他小功能,提高了开发人员的工作效率,包括编写和读取代码:

  • 目标类型 new 表达式
  • static 匿名函数
  • 目标类型的条件表达式
  • 扩展 GetEnumerator() 支持 foreach 循环
  • Lambda 表达式可以声明弃元参数
  • 特性可应用于本地函数

C# 9.0 版本继续致力于让 C# 成为一种新式通用编程语言。 功能继续支持新式工作负载和应用程序类型。

文章最初发布在 NDepend 博客上 ,由 Erik Dietrich 和 Patrick Smacchia 提供