运算符

?? 和 ??= 运算符 - Null 合并操作符

如果左操作数的值不为 null,则 null 合并运算符 ?? 返回该值;否则,它会计算右操作数并返回其结果。 如果左操作数的计算结果为非 null,则 ?? 运算符不会计算其右操作数。 仅当左操作数的计算结果为 null 时,Null 合并赋值运算符 ??= 才会将其右操作数的值赋值给其左操作数。 如果左操作数的计算结果为非 null,则 ??= 运算符不会计算其右操作数。

??= 运算符的左操作数必须是变量、属性索引器元素。

????= 运算符的左操作数的类型必须是可以为 null 的值类型。 特别是,可以使用具有无约束类型参数的 null 合并运算符:

1
2
3
private static void Display<T>(T a, T backup){
Console.WriteLine(a ?? backup);
}

null 合并运算符是右结合运算符。 也就是说,是窗体的表达式

C#复制

1
2
a ?? b ?? c
d ??= e ??= f

会像这样求值

C#复制

1
2
a ?? (b ?? c)
d ??= (e ??= f)

示例

????= 运算符在以下应用场景中很有用:

  • 在包含 null 条件运算符 ?.?[\] 的表达式中,当包含 null 条件运算的表达式结果为 null 时,可以使用 ?? 运算符来提供替代表达式用于求值:

    C#复制运行

    1
    2
    3
    4
    5
    6
    7
    double SumNumbers(List<double[]> setsOfNumbers, int indexOfSetToSum)
    {
    return setsOfNumbers?[indexOfSetToSum]?.Sum() ?? double.NaN;
    }

    var sum = SumNumbers(null, 0);
    Console.WriteLine(sum); // output: NaN
  • 当使用可以为 null 值类型并且需要提供基础值类型的值时,可以使用 ?? 运算符指定当可以为 null 的类型的值为 null 时要提供的值:

    1
    2
    3
    int? a = null;
    int b = a ?? -1;
    Console.WriteLine(b); // output: -1

    如果可以为 null 的类型的值为 null 时要使用的值应为基础值类型的默认值,请使用 Nullable.GetValueOrDefault() 方法。

  • 可以使用 throw 表达式作为 ?? 运算符的右操作数,以使参数检查代码更简洁:

    1
    2
    3
    4
    5
    public string Name
    {
    get => name;
    set => name = value ?? throw new ArgumentNullException(nameof(value), "Name cannot be null");
    }

    前面的示例还演示了如何使用 expression-bodied 成员来定义属性。

  • 可以使用 ??= 运算符替换窗体的代码

    1
    2
    3
    4
    if (variable is null)
    {
    variable = expression;
    }

    替换为以下代码:

    1
    variable ??= expression;