C#9.0 新特性简介

CandidateFeaturesForCSharp9

看到标题,是不是认为我把标题写错了?是的,C# 8.0还未正式发布,在官网它的最新版本还是Preview 5,通往C#9的漫长道路却已经开始.前写天收到了活跃在C#一线的BASSAM ALUGILI给我分享C# 9.0新特性,我在他文章的基础上进行翻译,希望能对大家有所帮助.

这是世界上第一篇关于C#9候选功能的文章。阅读完本文后,你将会为未来可能遇到的C# 9.0新特性做好更充分的准备。

这篇文章基于,

C# 9.0候选新特性

原生大小的数字类型

这次引入一组新类型(nint,nuint,nfloat等)'n'表示native(原生),该特性允许声明一个32位或64位的数据类型,这取决于操作系统的平台类型。

nint nativeInt = 55; take 4 bytes when I compile in 32 Bit host.
nint nativeInt = 55; take 8 bytes when I compile in 64 Bit host with x64 compilation settings. 

xamarin中已存在类似的概念,

xamarin原生类型

Records and Pattern-based With-Expression

这个功能我等待了很长时间,Records是一种轻量级的不可变类型,它可以是方法,属性,运算符等,它允许我们进行结构的比较, 此外,默认情况下,Records属性是只读的。

Records可以是值类型或引用类型。

Example

public class Point3D(double X, double Y, double Z);
public class Demo
{
 public void CreatePoint()
 {
 var p = new Point3D(1.0, 1.0, 1.0);
 }
} 

这些代码会被编译器转化如下形式.

public class Point3D
{
private readonly double <X>k__BackingField;
private readonly double <Y>k__BackingField;
private readonly double <Z>k__BackingField;
public double X {get {return <X>k__BackingField;}}
public double Y{get{return <Y>k__BackingField;}}
public double Z{get{return <Z>k__BackingField;}}  

 public Point3D(double X, double Y, double Z)
 {
 <X>k__BackingField = X;
 <Y>k__BackingField = Y;
 <Z>k__BackingField = Z;
 }  

 public bool Equals(Point3D value)
 {
 return X == value.X && Y == value.Y && Z == value.Z;
 }  

 public override bool Equals(object value)
 {
 Point3D value2;
 return (value2 = (value as Point3D)) != null && Equals(value2);
 }  

 public override int GetHashCode()
 {
 return ((1717635750 * -1521134295 + EqualityComparer<double>.Default.GetHashCode(X)) * -1521134295 + EqualityComparer<double>.Default.GetHashCode(Y)) * -1521134295 + EqualityComparer<double>.Default.GetHashCode(Z);
 }
}  

Using Records:  

public class Demo
{
 public void CreatePoint()
 {
 Point3D point3D = new Point3D(1.0, 1.0, 1.0);
 }
} 

Records迎合了基于表达式形式编程的特性,使得我们可以这样使用它.

var newPoint3D = Point3D.With(x: 42); 

这样我们创建的新Point(new Point3D)就像现有的一个(point3D)一样并把X的值更改为42。

这个特性于基于pattern matching也非常有效,我会在我的下一篇文章中介绍这一点.

那么我们为什么要使用Records而不是用结构体呢?为了回答这些问题,我引用了了Reddit的一句话:

“结构体是你必须要有一些约定来实现的东西。你不必手动地去让它只读,你也不用去实现他们的比较逻辑,但如果你不这样做,那你就失去了使用结构体的意义,编译器不会强制执行这些约束"。

Records类型由是编译器实现,这意味着您必须满足所有这些条件并且不能错误, 因此,它们不仅可以减少重复代码,还可以消除一大堆潜在的错误。

此外,这个功能在F#中存在了十多年,其他语言如(Scala,Kotlin)也有类似的概念。

F#

type Greeter(name: string) = member this.SayHi() = printfn "Hi, %s" name 

Scala

class Greeter(name: String)
{
  def SayHi() = println("Hi, " + name)
} 

Kotlin

class Greeter(val name: String)
{
 fun sayhi()
 {
 println("Hi, ${name}");
 }
}

在没有Records之前,我们要实现类似的功能,C#代码要这么写

C#

public class Greeter
{
 private readonly string _name;
 public Greeter(string name)
 {
 _name = name;
 }
 public void Greet()
 {
 Console.WriteLine($ "Hello, {_name}");
 }
}

有了Records之后,我们可以将C#代码大大地减少了,

ublic class Greeter(name: string)
{
 public void Greet()
 {
 Console.WriteLine($ "Hello, {_name}");
 }
} 

Less code! = I love it!

Type Classes

此功能的灵感来自Haskell,它是我最喜欢的功能之一。正如我两年前在我文章中所说,C#将实现更多的函数式编(FP)程概念,Type Classes就是FP概念之一。在函数式编程中,Type Classes允许您在类型上添加一组操作,但不实现它。由于实现是在其他地方完成的,这是一种多态,它比面向对象编程语言中的class更灵活。

Type Classes和C#接口具有相似的用途,但它们的工作方式有所不同,在某些情况下,由于处理固定类型而不是继承层次结构,因此Type Classes更易于使用。

此这特性最初与“extending everything”功能一起引入,您可以将它们组合在一起,如Mads Torgersen给出的例子所示。

我引用了官方提案中的一些结论:

“一般来说,”shape“(shape是Type Classes的一个新的关键字)声明非常类似于接口声明,除了以下情况,

  • 它可以定义任何类型的成员(包括静态成员)
  • 可以通过扩展实现
  • 只能在指定的地方当作一种类型使用(作用域)“

Haskell中 Type Classes示例。

class Eq a where
(==) :: a -> a -> Bool
(/=) :: a -> a -> Bool 

“Eq”是类名,而==,/ =是类中的操作。类型“a”是类“Eq”的实例。

如果我们将上述例子用C#接口实现将会是这样.

interface Num<A>
{
 A Add(A a, A b);
 A Mult(A a, A b);
 A Neg(A a);
}  

struct NumInt : Num<int>
{
 public int Add(int a, int b) => a + b;
 public int Mult(int a, int b) => a * b;
 public int Neg(int a) => -a;
} 

如果我们用Type Classes实现C# 功能会是这样

shape Num<A>
{
 A Add(A a, A b);
 A Mult(A a, A b);
 A Neg(A a);
}  

instance NumInt : Num<int>
{
 int Add(int a, int b) => a + b;
 int Mult(int a, int b) => a * b;
 int Neg(int a) => -a;
} 

通过上面例子,可以看到接口和shape的语法类似 ,那我们再来看看Mads Torgersen给出的例子

Note:shape不是一种类型。相反,shape的主要目的是用作通用约束,限制类型参数以具有正确的形状,同时允许通用声明的主体使用该形状,

原始来源

public shape SGroup<T>
{
 static T operator +(T t1, T t2);
 static T Zero {get;}
}

这个声明说如果一个类型在T上实现了一个+运算符并且具有0静态属性,那么它可以是一个SGroup 。

给int添加静态成员Zero

public extension IntGroup of int: SGroup<int>
{
 public static int Zero => 0;
} 

定义一个AddAll方法

public static AddAll<T>(T[] ts) where T: SGroup<T> // shape used as constraint
{
 var result = T.Zero; // Making use of the shape's Zero property
 foreach (var t in ts) { result += t; } // Making use of the shape's + operator
 return result;
} 

让我们用一些整数调用AddAll方法,

int[] numbers = { 5, 1, 9, 2, 3, 10, 8, 4, 7, 6 };
WriteLine(AddAll(numbers)); // infers T = int 

这就是Type class 的妙处,慢慢消化感受一下??

Dictionary Literals

引入更简单的语法来创建初始化的Dictionary <TKey,TValue>对象,而无需指定Dictionary类型名称或类型参数。使用用于数组类型推断的现有规则推断字典的类型参数。

// C# 1..8
var x = new Dictionary <string,int> () { { "foo", 4 }, { "bar", 5 }};
// C# 9
var x = ["foo":4, "bar": 5]; 

该特性使C#中的字典工作更简单,并删除冗余代码。此外,值得一提的是,在F#和Swift等其他编程语言中也使用了类似的字典语法。

Params Span

允许params语法使用Span 这个帮助来实现没有任何堆分配的params参数传递。此功能可以使params方法的使用更加高效。

新的语法如下,

void Foo(params Span<int> values); 

struct允许使用无参构造函数

到目前为止,在C#中不允许在结构体声明中使用无参构造函数,在C#9中,将删除此限制。

StackOverflow示例

public struct Rational
{
 private long numerator;
 private long denominator;  

 public Rational(long num, long denom)
 { /* Todo: Find GCD etc. */ }  

 public Rational(long num)
 {
 numerator = num;
 denominator = 1;
 }  

 public Rational() // This is not allowed
 {
 numerator = 0;
 denominator = 1;
 }
} 

链接到StackOverflow示例

其实CLR已经允许值类型数据具有无参构造函数,只是C# 对这个功能进行了限制,在C# 9.0中可能会消除这种限制.

固定大小的缓冲区

这些提供了一种通用且安全的机制,用于向C#语言声明固定大小的缓冲区。

目前,用户可以在不安全的环境中创建固定大小的缓冲区。但是,这需要用户处理指针,手动执行边界检查,并且只支持一组有限的类型(bool,byte,char,short,int,long,sbyte,ushort,uint,ulong,float和double)。该特性引入后将使固定大小的缓冲区变得安全安全,如下例所示。

可以通过以下方式声明一个安全的固定大小的缓冲区,

public fixed DXGI_RGB GammaCurve[1025]; 

该声明将由编译器转换为内部表示,类似于以下内容,

[FixedBuffer(typeof(DXGI_RGB), 1024)]
public ConsoleApp1.<Buffer>e__FixedBuffer_1024<DXGI_RGB> GammaCurve;  

// Pack = 0 is the default packing and should result in indexable layout.
[CompilerGenerated, UnsafeValueType, StructLayout(LayoutKind.Sequential, Pack = 0)]
struct <Buffer>e__FixedBuffer_1024<T>
{
 private T _e0;
 private T _e1;
 // _e2 ... _e1023
 private T _e1024;
 public ref T this[int index] => ref (uint)index <= 1024u ?
 ref RefAdd<T>(ref _e0, index):
 throw new IndexOutOfRange();
} 

Uft8字符串文字

它是关于定义一种新的字符串类型UTF8String,它将是,

System.UTF8String myUTF8string ="Test String"; 

Base(T)

此功能用于解决默认接口方法中的覆盖冲突问题:

interface I1
{
  void M(int) { }
}

interface I2
{
  void M(short) { }
}

interface I3
{
  override void I1.M(int) { }
}

interface I4 : I3
{
  void M2()
  {
    base(I3).M(0) // Which M should be used here? What does this do?
  }
}

更多信息,

https://github.com/dotnet/csharplang/issues/2337

https://github.com/dotnet/csharplang/blob/master/meetings/2019/LDM-2019-02-27.md

摘要

您已经阅读了第一个C#9候选特性。正如您所看到的,许多新功能受到其他编程语言或编程范例的启发,而不是自我创新,这些特性大部分在在社区中得到了广泛认可,所以引入C# 后应该也会给大家带来不错的体验.

原文 : https://www.c-sharpcorner.com/article/candidate-features-for-c-sharp-9/

到此这篇关于C#9.0 新特性简介的文章就介绍到这了,更多相关C#9.0 新特性内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • C# 9.0 新特性之模式匹配简化的实现

    记得在 MS Build 2020 大会上,C# 语言开发项目经理 Mads Torgersen 宣称 C# 9.0 将会随着 .NET 5 在今年 11 月份正式发布.目前 .NET 5 已经到了 Preview 5 阶段了,C# 9.0 也已经初具规模.忍不住激动的心情,暂停更新<C#.NET 拾遗补漏>系列几天,先要和大家分享一下我了解到的 C# 9.0 的新特性.由于新特性比较多,所以会分成几篇来讲.这是第一篇,专讲模式匹配这个特性的简化. 模式匹配(Pattern Matching)

  • 浅析C# 9.0 新特性之 Lambda 弃元参数

    大家好,这是 C# 9.0 新特性短系列的第 5 篇文章. 弃元(Discards) 是在 C# 7.0 的时候开始支持的,它是一种人为丢弃不使用的临时虚拟变量.语法上它是用来赋值的,但它却不被分配存储空间,即没有值,所以不能从中读取值.弃元用 _(下划线) 表示,下划线是一个关键字,只能赋值,不能读取,例如: 在 C# 7.0 中,弃元的使用场景主要有下面四种: 元组和对象的解构 使用 is 和 switch 的模式匹配 对具有 out 参数的方法的调用 作用域内独立使用场景 针对这几个场景,

  • 浅谈C#9.0新特性之参数非空检查简化

    参数非空检查是缩写类库很常见的操作,在一个方法中要求参数不能为空,否则抛出相应的异常.比如: public static string HashPassword(string password) { if(password is null) { throw new ArgumentNullException(nameof(password)); } ... } 当异常发生时,调用者很容易知道是什么问题.如果不加这个检查,可能就会由系统抛出未将对象引用为实例之类的错误,这不利于调用者诊断错误. 由

  • 浅谈C# 9.0 新特性之只读属性和记录

    大家好,这是 C# 9.0 新特性系列的第 4 篇文章. 熟悉函数式编程的童鞋一定对"只读"这个词不陌生.为了保证代码块自身的"纯洁",函数式编程是不能随便"弄脏"外来事物(参数.变量等)的,所以"只读"对函数式编程非常重要. 为了丰富 C# 对函数式编程支持,较新的 C# 版本引入了一些很有用的新特性.比如 C# 8 中就对 struct 类型的方法增加了 readonly 修饰符支持,被 readonly 修饰的方法是不能

  • c# 9.0新特性nint和Pattern matching的使用方法

    一:背景 1. 讲故事 上一篇跟大家聊到了Target-typed new 和 Lambda discard parameters,看博客园和公号里的阅读量都达到了新高,甚是欣慰,不管大家对新特性是多头还是空头,起码还是对它抱有一种极为关注的态度,所以我的这个系列还得跟,那就继续开撸吧,今天继续带来两个新特性,更多新特性列表,请大家关注:新特性预览 二:新特性研究 1. Native ints 从字面上看貌似是什么原生类型ints,有点莫名其妙,还是看一看Issues上举得例子吧: Summar

  • C#9.0 新特性简介

    CandidateFeaturesForCSharp9 看到标题,是不是认为我把标题写错了?是的,C# 8.0还未正式发布,在官网它的最新版本还是Preview 5,通往C#9的漫长道路却已经开始.前写天收到了活跃在C#一线的BASSAM ALUGILI给我分享C# 9.0新特性,我在他文章的基础上进行翻译,希望能对大家有所帮助. 这是世界上第一篇关于C#9候选功能的文章.阅读完本文后,你将会为未来可能遇到的C# 9.0新特性做好更充分的准备. 这篇文章基于, C# 9.0候选新特性 原生大小的

  • MySQL 8.0新特性 — 管理端口的使用简介

    前言 下面这个报错,相信大多数童鞋都遇见过:那么碰到这个问题,我们应该怎么办呢?在MySQL 5.7及之前版本,出现"too many connection"报错,超级用户root也无法登录上去,除了重启实例,没有其他更好的解决办法:不过在MySQL 8.0版本中,是对连接管理做了一些优化,下面我们就来看一下. ERROR 1040 (HY000): Too many connections 连接管理 在MySQL 8.0版本中,对连接管理这一块,是先后做了两个比较大的改变:一个是允许

  • MySQL 8.0新特性 — 检查性约束的使用简介

    前言 在MySQL 8.0版本中,引入了一个非常有用的新特性 - 检查性约束,它可以提高对非法或不合理数据写入的控制能力:接下来我们就来详细了解一下. 检查性约束 创建.删除与查看 (1)可以在建表时,创建检查性约束 mysql> CREATE TABLE t1 -> ( -> CHECK (c1 <> c2), -> c1 INT CHECK (c1 > 10), -> c2 INT CONSTRAINT c2_positive CHECK (c2 >

  • C# 8.0新特性介绍

    C# 语言是在2000发布的,至今已正式发布了7个版本,每个版本都包含了许多令人兴奋的新特性和功能更新.同时,C# 每个版本的发布都与同时期的 Visual Studio 以及 .NET 运行时版本高度耦合,这也有助于开发者更好的学习掌握 C#,并将其与 Visual Studio 以及 .NET 的使用结合起来. 加快 C# 版本的发布速度 在被称为"新微软"的推动下,微软创新的步伐也加快了.为了做到加快步伐,微软开发部门将一些过去集成在一起的技术现在都分离了出来. Visual S

  • AngularJS 2.0新特性有哪些

    AngularJS已然成为Web应用开发世界里最受欢迎的开源JavaScript框架.自成立以来,见证其成功的是惊人的经济增长以及团体的支持与采用--包括个人开发者.企业.社区. Angular已经变成一个构建复杂单页面应用的客户端MVW框架(Model-View-Whatever).它在应用测试和应用编写方面都扮演重要角色,同时简化了开发过程. Angular目前的版本为1.3,该版本稳定,并被谷歌(框架维护者)用于支持众多应用(据估计,在谷歌有超过1600个应用运行于Angular1.2或1

  • C# 6.0 新特性汇总

    1. 静态using(static using) 静态using声明允许不使用类名直接调用静态方法. The static using declaration allows invoking static methods without the class name. In C# 5 using System; Console.WriteLine("Hello, World!"); In C# 6 using static System.Console; WriteLine("

  • PHP 5昨天隆重推出--PHP 5/Zend Engine 2.0新特性

    前言 今天突然想到PHP官方网站上一转,一眼就看到PHP5推出的通告.虽然以前看到过PHP5的预告,但还是仔细看了PHP 5/Zend Engine 2.0新特性一文,一股JAVA气息扑面而来...   特将该文试译出来,首发于CSDN网站,以飨读者. PHP 5/Zend Engine 2.0新特性徐唤春 译 sfwebsite@hotmail.comhttp://www.php.net/zend-engine-2.php 全新的对象模型PHP中的对象处理部分已完全重写,具有更佳的性能和更多的

  • Android Studio3.0新特性及安装图文教程

    Android Studio是Android的官方IDE.它是专为Android而打造,可以加快您的开发速度,帮助您为每款Android设备构建最优应用. 它提供专为Android开发者量身定制的工具,其中包括丰富的代码编辑.调试.测试和性能分析工具. 一.Android Studio3.0新特性 (1).核心IDE更改 我们将基础IDE从IntelliJ 2016.2升级到2017.1.2,在2016.3和 2017.1中增加了许多新功能, 包括参数提示,语义突出显示,搜索中的即时结果等等.

  • Android5.0新特性详解之全新的动画

    在Material Design设计中,为用户与app交互反馈他们的动作行为和提供了视觉上的连贯性.Material主题为控件和Activity的过渡提供了一些默认的动画,在android L上,允许自定义这些动画: Touch feedback 触摸反馈 Circular Reveal 圆形展示 Curved motion 曲线运动 View state changes 视图状态变化 Vector Drawables 矢量图动画 Activity transitions 活动转场 触摸反馈 触

  • Angular5.0.0新特性

    文章来自官网部分翻译 https://blog.angular.io/version-5-0-0-of-angular-now-available-37e414935ced Angular5.0.0版本已经正式发布 总结一下v5.0.0带来的新变化都有哪些. 1.构建优化 5.0版本默认采用CLI构建和打包.构建优化器是包含在CLI里面的一个工具,通过对你的应用程序更加语义化的理解可以使得你的打包程序(bundle)更小. 构建优化器有两个主要工作. 第一,我们可以将应用程序的一部分标记为纯应用

随机推荐