合规国际互联网加速 OSASE为企业客户提供高速稳定SD-WAN国际加速解决方案。 广告
# 使用可以为 null 的类型(C# 编程指南) 可以为 null 的类型可以表示基础类型的所有值,另外还可以表示 [null](https://msdn.microsoft.com/zh-cn/library/edakx9da.aspx) 值。可以为 null 的类型可通过下面两种方式中的一种声明: System.Nullable&lt;T&gt; variable \- 或 \- T?variable T 是可以为 null 的类型的基础类型。 T 可以是包括 **struct** 在内的任何值类型;但不能是引用类型。 有关可能使用可以为 null 的类型的示例,请考虑普通的布尔变量如何能够具有两个值:true 和 false。不存在表示“未定义”的值。在很多编程应用中(最突出的是数据库交互),变量可以以未定义的状态出现。例如,数据库中的某个字段可能包含值 true 或 false,但是它也可能根本不包含值。同样,可以将引用类型设置为 **null**,以指示它们未初始化。 这种不一致会导致额外的编程工作,如使用附加变量来存储状态信息、使用特殊值,等等。可以为 null 的类型修饰符使 C# 能够创建表示未定义值的值类型变量。 ## 可以为 null 的类型示例 任何值类型都可用作可以为 null 的类型的基础。例如: ``` int? i = 10; double? d1 = 3.14; bool? flag = null; char? letter = 'a'; int?[] arr = new int?[10]; ``` ## 可以为 null 的类型的成员 可以为 null 的类型的每个实例都具有两个公共的只读属性: * **HasValue** HasValue 属于 **bool** 类型。当变量包含非 null 值时,它被设置为 **true**。 * **Value** Value 的类型与基础类型相同。如果 HasValue 为 **true**,则说明 Value 包含有意义的值。如果 HasValue 为 **false**,则访问 Value 将引发 [InvalidOperationException](https://msdn.microsoft.com/zh-cn/library/system.invalidoperationexception.aspx)。 在此示例中,HasValue 成员用于在尝试显示变量之前测试它是否包含值。 ``` int? x = 10; if (x.HasValue) { System.Console.WriteLine(x.Value); } else { System.Console.WriteLine("Undefined"); } ``` 也可以如下面的示例所示对值进行测试: ``` int? y = 10; if (y != null) { System.Console.WriteLine(y.Value); } else { System.Console.WriteLine("Undefined"); } ``` ## 显式转换 可以为 null 的类型可强制转换为常规类型,方法是使用强制转换来显式转换或者通过使用 Value 属性来转换。例如: ``` int? n = null; //int m1 = n; // Will not compile. int m2 = (int)n; // Compiles, but will create an exception if n is null. int m3 = n.Value; // Compiles, but will create an exception if n is null. ``` 如果两种数据类型之间定义了用户定义的转换,则同一转换也可用于这些数据类型的可以为 null 的版本。 ## 隐式转换 可使用 **null** 关键字将可以为 null 的类型的变量设置为 null,如以下示例所示: ``` int? n1 = null; ``` 从普通类型到可以为 null 的类型的转换是隐式的。 ``` int? n2; n2 = 10; // Implicit conversion. ``` ## 运算符 可以为 null 的类型还可以使用预定义的一元和二元运算符,以及现有的任何用户定义的值类型运算符。如果操作数为 null,这些运算符将产生一个 null 值;否则运算符将使用包含的值来计算结果。例如: ``` int? a = 10; int? b = null; a++; // Increment by 1, now a is 11. a = a * 10; // Multiply by 10, now a is 110. a = a + b; // Add b, now a is null. ``` 在对可以为 null 的类型执行比较时,如果其中一个可以为 null 的类型的值为 null,但另外一个类型的值不为 null,则除 **!=**(不等于)外,所有比较的结果都将为 **false**。一定不要以为由于一个特定比较的结果为 **false**,相反的情况就会为 **true**。在以下示例中,10 不大于、小于或等于 null。只有 num1 != num2 的计算结果为 **true**。 ``` int? num1 = 10; int? num2 = null; if (num1 >= num2) { Console.WriteLine("num1 is greater than or equal to num2"); } else { // This clause is selected, but num1 is not less than num2. Console.WriteLine("num1 >= num2 returned false (but num1 < num2 also is false)"); } if (num1 < num2) { Console.WriteLine("num1 is less than num2"); } else { // The else clause is selected again, but num1 is not greater than // or equal to num2. Console.WriteLine("num1 < num2 returned false (but num1 >= num2 also is false)"); } if (num1 != num2) { // This comparison is true, num1 and num2 are not equal. Console.WriteLine("Finally, num1 != num2 returns true!"); } // Change the value of num1, so that both num1 and num2 are null. num1 = null; if (num1 == num2) { // The equality comparison returns true when both operands are null. Console.WriteLine("num1 == num2 returns true when the value of each is null"); } /* Output: * num1 >= num2 returned false (but num1 < num2 also is false) * num1 < num2 returned false (but num1 >= num2 also is false) * Finally, num1 != num2 returns true! * num1 == num2 returns true when the value of each is null */ ``` 如果两个可以为 null 的类型的值均为 null,则其相等比较的计算结果为 **true**。 ## ??运算符 **??** 运算符定义在将可以为 null 的类型分配给非可以为 null 的类型时返回的默认值。 ``` int? c = null; // d = c, unless c is null, in which case d = -1. int d = c ?? -1; ``` 此运算符还可用于多个可以为 null 的类型。例如: ``` int? e = null; int? f = null; // g = e or f, unless e and f are both null, in which case g = -1. int g = e ?? f ?? -1; ``` ## bool?type **bool?** 可以为 null 的类型可以包含三个不同的值:[true](https://msdn.microsoft.com/zh-cn/library/eahhcxk2.aspx)、[false](https://msdn.microsoft.com/zh-cn/library/67bxt5ee.aspx) 和 [null](https://msdn.microsoft.com/zh-cn/library/edakx9da.aspx)。有关如何从 bool? 强制转换为 bool 的信息,请参见[如何:安全地将 bool? 强制转换为 bool(C# 编程指南)](https://msdn.microsoft.com/zh-cn/library/bb384091.aspx)。 可以为 null 的布尔值类似于 SQL 中使用的布尔变量类型。若要确保由 **&** 和 产生的结果**|** 运算符与 SQL 中的三值布尔类型一致,提供了以下预定义的运算符: bool?operator &(bool?x, bool?y) bool?operator |(bool?x, bool?y) 下表中列出了这些运算符的结果: | X | y | x&y | x&#124;y | | --- | --- | --- | --- | | true | true | true | true | | true | false | false | true | | true | null | null | true | | false | true | false | true | | false | false | false | false | | false | null | false | null | | null | true | null | true | | null | false | false | null | | null | null | null | null | ## 请参阅 [C# 编程指南](https://msdn.microsoft.com/zh-cn/library/67ef8sbd.aspx) [可以为 null 的类型(C# 编程指南)](https://msdn.microsoft.com/zh-cn/library/1t3y8s4s.aspx) [装箱可以为 null 的类型(C# 编程指南)](https://msdn.microsoft.com/zh-cn/library/ms228597.aspx) [可以为 Null 的值类型 (Visual Basic)](https://msdn.microsoft.com/zh-cn/library/ms235245.aspx)