다음을 통해 공유


Null 비교

데이터 소스의 null 값은 값을 알 수 없음을 나타냅니다. LINQ to Entities 쿼리에서는 특정 계산이나 비교가 유효한 데이터 또는 Null이 아닌 데이터를 가진 행에서만 수행되도록 Null 값을 확인할 수 있습니다. 그러나 CLR Null 의미 체계는 데이터 소스의 Null 의미 체계와 다를 수 있습니다. 대부분의 데이터베이스는 세 개의 값으로 구성된 논리 버전을 사용하여 Null 비교를 처리합니다. 즉, Null 값에 대한 비교는 true 또는 false가 되지 않고 unknown이 됩니다. ANSI Null은 이렇게 구현되는 경우가 많지만 항상 그런 것은 아닙니다.

기본적으로 SQL Server에서 Null은 Null과 같음 비교는 Null 값을 반환합니다. 다음 예제에서는 ShipDate가 Null인 행은 결과 집합에서 제외되고 Transact-SQL 문에서 0개 행을 반환합니다.

-- Find order details and orders with no ship date.  SELECT h.SalesOrderID
FROM Sales.SalesOrderHeader h
JOIN Sales.SalesOrderDetail o ON o.SalesOrderID = h.SalesOrderID
WHERE h.ShipDate IS Null  

Null은 Null과 같음 비교에서 true를 반환하는 CLR Null 의미 체계와 이 동작은 전혀 다릅니다.

다음 LINQ 쿼리는 CLR로 표현되지만 데이터 소스에서 실행됩니다. CLR 의미 체계가 데이터 소스에서 반영될 것이라는 보장이 없으므로 예상 동작을 결정할 수 없습니다.

Using context As New AdventureWorksEntities()

    Dim orders As ObjectSet(Of SalesOrderHeader) = context.SalesOrderHeaders
    Dim details As ObjectSet(Of SalesOrderDetail) = context.SalesOrderDetails

    Dim query = _
        From order In orders _
        Join detail In details _
        On order.SalesOrderID _
        Equals detail.SalesOrderID _
        Where order.ShipDate = Nothing
        Select order.SalesOrderID


    For Each orderID In query
        Console.WriteLine("OrderID: {0} ", orderID)
    Next
End Using
using (AdventureWorksEntities context = new AdventureWorksEntities())
{
    ObjectSet<SalesOrderHeader> orders = context.SalesOrderHeaders;
    ObjectSet<SalesOrderDetail> details = context.SalesOrderDetails;

    var query =
        from order in orders 
        join detail in details 
        on order.SalesOrderID 
        equals detail.SalesOrderID 
        where order.ShipDate == null
        select order.SalesOrderID;

    foreach (var OrderID in query)
    {
        Console.WriteLine("OrderID : {0}", OrderID);
    }
}

키 선택기

키 선택기는 요소에서 키를 추출하기 위해 표준 쿼리 연산자에서 사용된 함수입니다. 키 선택기 함수에서 식을 상수와 비교할 수 있습니다. 식을 Null 상수와 비교하거나 두 개의 Null 상수를 비교하면 CLR Null 의미 체계가 표시됩니다. 데이터 소스에서 Null 값을 가진 두 개의 열을 비교하면 저장소 Null 의미 체계가 표시됩니다. 키 선택기는 GroupBy과 같은 대부분의 그룹화 및 정렬 표준 쿼리 연산자에서 발견되며, 쿼리 결과를 정렬하거나 그룹화하는 기준 키를 선택하는 데 사용됩니다.

Null 개체의 Null 속성

Entity Framework 에서 Null 개체의 속성은 Null입니다. CLR에서 Null 개체의 속성을 참조하려고 하면 NullReferenceException이 발생합니다. LINQ 쿼리에 Null 개체의 속성이 필요한 경우 일관성 없는 동작이 발생할 수 있습니다.

예를 들어, 다음 쿼리에서 NewProduct로의 캐스팅은 명령 트리 계층에서 수행되며, 이로 인해 Introduced 속성이 Null이 될 수 있습니다. DateTime 비교가 true가 되도록 데이터베이스에서 Null 비교를 정의한 경우 해당 행이 포함됩니다.

Using context As New AdventureWorksEntities()
    Dim dt As DateTime = New DateTime()
    Dim query = context.Products _
        .Where(Function(p) _
            ((DirectCast(p, NewProduct)).Introduced > dt)) _
        .Select(Function(x) x)
End Using
using (AdventureWorksEntities context = new AdventureWorksEntities())
{

    DateTime dt = new DateTime();
    var query = context.Products
        .Where(p => (p as NewProduct).Introduced > dt)
        .Select(x => x);
}

Null 컬렉션을 집계 함수에 전달

LINQ to Entities 에서 IQueryable을 지원하는 컬렉션을 집계 함수로 전달하면 데이터베이스에서 집계 작업이 수행됩니다. 메모리 내에서 수행된 쿼리의 결과와 데이터베이스에서 수행된 쿼리의 결과가 서로 다를 수 있습니다. 메모리 내 쿼리에서 일치하는 항목이 없으면 쿼리가 0을 반환합니다. 데이터베이스에서 동일한 쿼리가 null을 반환합니다. null 값이 LINQ 집계 함수로 전달되면 예외가 throw됩니다. 가능한 null 값을 받아들이려면 형식 및 쿼리 결과를 받는 형식의 속성을 nullable 형식으로 캐스팅합니다.

참고 항목

개념

LINQ to Entities 쿼리의 식