?? 및?? = 연산자 - null 병합 연산자

null 병합 연산자 ??null이 아닌 경우 왼쪽 피연산자의 값을 반환합니다. 그렇지 않으면 오른쪽 피연자를 평가하고 그 결과를 반환합니다. 왼쪽 피연산자가 null이 아닌 것으로 평가되면 ?? 연산자는 오른쪽 피연산자를 평가하지 않습니다. null 병합 할당 연산 ??= 자는 왼쪽 피연산자가 로 계산되는 경우에만 오른쪽 피연산자의 값을 왼쪽 피연산자에 할당합니다.null 왼쪽 피연산자가 null이 아닌 것으로 평가되면 ??= 연산자는 오른쪽 피연산자를 평가하지 않습니다.

List<int> numbers = null;
int? a = null;

Console.WriteLine((numbers is null)); // expected: true
// if numbers is null, initialize it. Then, add 5 to numbers
(numbers ??= new List<int>()).Add(5);
Console.WriteLine(string.Join(" ", numbers));  // output: 5
Console.WriteLine((numbers is null)); // expected: false        


Console.WriteLine((a is null)); // expected: true
Console.WriteLine((a ?? 3)); // expected: 3 since a is still null 
// if a is null then assign 0 to a and add a to the list
numbers.Add(a ??= 0);
Console.WriteLine((a is null)); // expected: false        
Console.WriteLine(string.Join(" ", numbers));  // output: 5 0
Console.WriteLine(a);  // output: 0	        

??= 연산자의 왼쪽 피연산자는 변수, 속성 또는 인덱서 요소여야 합니다.

??= 연산자의 왼쪽 피연산자 ?? 형식은 nullable이 아닌 값 형식일 수 없습니다. 특히 제약이 없는 형식 매개 변수와 함께 null 병합 연산자를 사용할 수 있습니다.

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

null 병합 연산자는 오른쪽 결합입니다. 즉, 양식의 식이

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

다음과 같이 계산됩니다.

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

????= 연산자는 다음과 같은 시나리오에서 유용할 수 있습니다.

  • null 조건부 연산 ?. 자 및 ?[]가 있는 식에서는 연산자를 사용하여 ?? null 조건부 연산이 있는 식의 결과가 인 경우를 계산하는 대체 식을 제공할 수 있습니다.null

    double SumNumbers(List<double[]> setsOfNumbers, int indexOfSetToSum)
    {
        return setsOfNumbers?[indexOfSetToSum]?.Sum() ?? double.NaN;
    }
    
    var sum = SumNumbers(null, 0);
    Console.WriteLine(sum);  // output: NaN
    
  • nullable 값 형식을 사용하고 기본값 유형의 값을 제공해야 할 때 ?? 연산자를 사용하여 nullable 값이 null인 경우 제공할 값을 지정합니다.

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

    nullable 형식 값이 null일 때 사용될 값이 기본 값 형식의 기본 값이어야 하는 경우 Nullable<T>.GetValueOrDefault() 메서드를 사용합니다.

  • 연산자의 throw 오른쪽 피연산 ?? 자로 식을 사용하여 인수 검사 코드를 더 간결하게 만들 수 있습니다.

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

    앞의 예제에서는 식 본문 멤버를 사용하여 속성을 정의하는 방법도 보여줍니다.

  • 연산자를 ??= 사용하여 양식의 코드를 바꿀 수 있습니다.

    if (variable is null)
    {
        variable = expression;
    }
    

    다음 코드와 바꿉니다.

    variable ??= expression;
    

연산자 오버로드 가능성

연산 ?? 자 및 ??= 는 오버로드할 수 없습니다.

C# 언어 사양

?? 연산자에 대한 자세한 내용은 C# 언어 사양null 병합 연산자 섹션을 참조하세요.

??= 연산자에 대한 자세한 내용은 기능 제안 노트를 참조하세요.

참고 항목