可空类型修饰符(?)
引用类型可以使用空引用表示一个不存在的值,而值类型通常不能表示为空。
例如:string str=null;
是正确的,int i=null;
编译器就会报错。
为了使值类型也可为空,就可以使用可空类型,即用可空类型修饰符 “?“ 来表示,表现形式为 “T?”
例如:int?
表示可空的整形,DateTime?
表示可为空的时间。
T? 其实是 System.Nullable(泛型结构)的缩写形式,
也就意味着当你用到 T? 时编译器编译时会把 T? 编译成 System.Nullable 的形式。
例如:int?
,编译后便是 System.Nullable 的形式。
三元(运算符)表达式(?: )
例如:x?y:z
表示如果表达式 x 为 true
,则返回 y,
如果 x 为 false
,则返回 z,是省略 if{}else{}
的简单形式。
空合并运算符(??)
用于定义可空类型和引用类型的默认值。
如果此运算符的左操作数不为 null,则此运算符将返回左操作数,否则返回右操作数。
例如:a??b
当 a 为 null 时则返回 b,a 不为 null
时则返回 a 本身。
空合并运算符为右结合运算符,即操作时从右向左进行组合的。
如,“a??b??c
” 的形式按 “a??(b??c)
” 计算。
NULL检查运算符(?.)
例如我们要获取一个 Point 序列的第一个点的 X 坐标,第一感觉会这么写:
1 | int firstX = points.First().X; |
但是,老鸟会告诉你,这儿没有进行 NULL 检查,正确的版本是这样的:
1 | int? firstX = null; |
正确倒是正确了,代码取变得难读多了。在 C# 6.0 中,引入了一个 ?.
的运算符,前面的代码可以改成如下形式:
1 | int? firstX = points?.FirstOrDefault()?.X; |
从这个例子中我们也可以看出它的基本用法:如果对象为 NULL,则不进行后面的获取成员的运算,直接返回 NULL。
需要注意的是,由于“?.
” 运算符返回的可以是 NULL,当返回的成员类型是 struct 类型的时候,?.
和 .
运算符的返回值类型是不一样的。
1 | Point p = new Point(3, 2); |
?[x]
运算符
1 | var first = customers?[0].Orders.Count(); |
(这个不太清楚,感觉微软语法糖用过头了。)
本质:都使用了语法糖,在编译阶段编译器会把特殊符号先转译成原始状态再编译。