Loading

C# 9.0 新特性预览 - 类型推导的 new

C# 9.0 新特性预览 - 类型推导的 new

前言

随着 .NET 5 发布日期的日益临近,其对应的 C# 新版本已确定为 C# 9.0,其中新增加的特性(或语法糖)也已基本锁定,本系列文章将向大家展示它们。

目录

[C# 9.0 新特性预览 - 类型推导的 new]
[C# 9.0 新特性预览 - 空参数校验]
[C# 9.0 新特性预览 - Lambda 中的弃元]
[C# 9.0 新特性预览 - Record 类型]
[C# 9.0 新特性预览 - 模式匹配的改善]
[C# 9.0 新特性预览 - 其他小的变化]

具有类型推导的 new 表达式 (Target-typed new expressions)

这是一个本应随着 C# 8.0 发布的语言特性,但因种种原因在发布 C# 8.0 的最后关头,它被移出了最终的发布版本,下面我们来认识认识它。
大家都知道,C# 在3.0中新增 var 关键字来做隐式类型声明,把繁重的声明语法简化了。

Dictionary<string, List<int>> field = new Dictionary<string, List<int>>()
// 可以简化为
var field = new Dictionary<string, List<int>>()

var 关键字的基本原理不再复述,简单说就是编译器可以根据等号后面的类型推导出 var 的类型,那么是不是也可以反过来,我们先声明类型,接下来的 new 关键字后面就不用写类型了呢?于是本文介绍的特性来了:

Dictionary<string, List<int>> field = new Dictionary<string, List<int>>()
// C# 9.0 中可以写成
Dictionary<string, List<int>> field = new()

从以上代码可以看出语法很简单,即省略了繁琐的可以推导出的类型。
其语法 Spec 如下:

'new' '(' argument_list? ')' object_or_collection_initializer?

搭配初始化器,我们可以进一步简化带有初始值的初始化。

Dictionary<string, List<int>> field = new() {
    { "item1", new() { 1, 2, 3 } }
};

进一步展示该语法在各种情况下的使用

在所有可以推导出类型的上下文中,都可以使用,例如:

XmlReader.Create(reader, new() { IgnoreWhitespace = true });

带有参数的构造方法:

class C {
    C(params int[] p) {}
}

C c = new(1, 2, 3);

调用方法时:

class A {}
static void M(A a) {};

M(a: new());

配合对象初始化器:

X x = new() { field = 42 };

泛型的类型推导,需要注意,要有一个显示类型声明才能正确推导:

void M<T>(T t1, T t2) {}

M(new X(), new());

类似的,数组的声明

var arr = new[] {new X(), new()};

不适用此特性的场景

值类型的初始化,可以使用 default 替代

int x = new(); // ERROR
Struct y = new(); // ERROR

使用 as 操作时无法正确推导

Console.Write(new() as X); // ERROR

自然,使用 var 时也无法推导

var x = new(); // ERROR

有歧义的重载

void M(object a, X b) => Console.Write($"{a} {b}");
void M(X a, object b) => Console.Write($"{a} {b}");

M(new(), new()); // ERROR

需要注意的地方

这个新语法 new(),比较容易与匿名类型语法混淆 new{},它们两个是完全不同的东西,需要注意一下。

参考

[Proposal: Target-typed new expressions]
[Unit test: TargetTypedNewTests.cs]

posted @ 2020-05-06 22:48  Rwing  阅读(7815)  评论(53编辑  收藏  举报