Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)

Entity Framework의 Join 사용 시 다중 칼럼에 대한 OR 조건 쿼리

다음과 같은 질문이 있군요.

안녕하세요! EntityFramework Linq 질문 드리겠습니다.
; https://www.sysnet.pe.kr/3/0/5558

간단하게 테스트를 위해 다음의 글을 참조해,

Tutorial: Get Started with Entity Framework 6 Code First using MVC 5
; https://docs.microsoft.com/en-us/aspnet/mvc/overview/getting-started/getting-started-with-ef-using-mvc/creating-an-entity-framework-data-model-for-an-asp-net-mvc-application

(위의 글은 EF를 사용했지만) Entity Framework Core를 이용한 콘솔 프로젝트를 만들어 보겠습니다. 우선 .NET Core/5+ 프로젝트를 만들고 참조를 2개 추가합니다.

Install-Package Microsoft.EntityFrameworkCore
Install-Package Microsoft.EntityFrameworkCore.SqlServer

그다음 엔티티 타입으로 Student와 Book을 추가하고,

using System.Collections.Generic;

namespace ConsoleApp1
{
    public class Student
    {
        public int ID { get; set; }
        public string Name { get; set; }
        public int SubID { get; set; } // JOIN OR 테스트를 위한 칼럼 추가

        public virtual ICollection<Book> Enrollments { get; set; }
    }

    public class Book
    {
        public int ID { get; set; }
        public int StudentID { get; set; }
        public string Title { get; set; }
        public int SubID { get; set; } // JOIN OR 테스트를 위한 칼럼 추가
    }
}

DbContext를 이렇게 구성해 주면 됩니다.

using Microsoft.EntityFrameworkCore;

namespace ConsoleApp1
{
    public class SchoolContext : DbContext
    {

        public SchoolContext() : base()
        {
        }

        public DbSet<Student> Students { get; set; }
        public DbSet<Book> Books { get; set; }

        protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
        {
            base.OnConfiguring(optionsBuilder);

            string cs = "Data Source=.;Initial Catalog=ContosoUniversity1;Integrated Security=SSPI;";
            optionsBuilder.UseSqlServer(cs);
        }
    }
}

마지막으로 다음과 같이 예제 데이터와 함께 DB를 생성하는 코드를 추가하면 준비가 완료됩니다. ^^

using Microsoft.EntityFrameworkCore;
using System;
using System.Collections.Generic;
using System.Linq;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            SchoolContext db = new SchoolContext();

            // EF 6는 https://www.entityframeworktutorial.net/code-first/database-initialization-strategy-in-code-first.aspx
            if (db.Database.CanConnect() == false)
            {
                db.Database.EnsureCreated();
                InsertSampleData(db);
            }
        }

        private static void InsertSampleData(SchoolContext context)
        {
            var students = new List<Student>
                       {
                           new Student{Name="Carson",SubID=1},
                           new Student{Name="Meredith",SubID=2},
                           new Student{Name="Arturo",SubID=3},
                           new Student{Name="Gytis",SubID=1},
                           new Student{Name="Yan",SubID=2},
                           new Student{Name="Peggy",SubID=3},
                           new Student{Name="Laura",SubID=1},
                           new Student{Name="Nino",SubID=2}
                       };

            students.ForEach(s => context.Students.Add(s));
            context.SaveChanges();
            var courses = new List<Book>
                       {
                           new Book{StudentID=1, Title="Book 1", SubID=1},
                           new Book{StudentID=1, Title="Book 1", SubID=1},
                           new Book{StudentID=1, Title="Book 1", SubID=2},
                           new Book{StudentID=2, Title="Book 2", SubID=2},
                           new Book{StudentID=2, Title="Book 2", SubID=3},
                           new Book{StudentID=2, Title="Book 2", SubID=3},
                           new Book{StudentID=3, Title="Book 3", SubID=4},
                           new Book{StudentID=4, Title="Book 4", SubID=4},
                           new Book{StudentID=4, Title="Book 4", SubID=5},
                           new Book{StudentID=5, Title="Book 5", SubID=5},
                           new Book{StudentID=6, Title="Book 6", SubID=6},
                           new Book{StudentID=7, Title="Book 7", SubID=6},
                       };
            courses.ForEach(s => context.Books.Add(s));
            context.SaveChanges();
        }

    }
}




자, 이제 본격적으로 INNER JOIN 테스트를 해볼까요? ^^ 다들 아시겠지만 기본적인 JOIN은 LINQ 쿼리로 다음과 같이 작성할 수 있습니다.

{
    var q = from s in db.Students
            join e in db.Books on s.ID equals e.StudentID
            select s;

    Console.WriteLine(q.ToQueryString());
}
/* 출력 결과
SELECT [s].[ID], [s].[Name], [s].[SubID]
FROM [Students] AS [s]
INNER JOIN [Books] AS [b] ON [s].[ID] = [b].[StudentID]     
*/

이 상태에서 JOIN의 AND 조건을 추가하는 것은 다음과 같이 on과 equals에 다중 AND 조건을 매칭시키는 방식으로 쉽게 구현할 수 있습니다.

{
    var q = from s in db.Students
            join e in db.Books on new { C1 = s.ID, C2 = s.SubID }
                           equals new { C1 = e.StudentID, C2 = e.SubID }
            select s;

    Console.WriteLine(q.ToQueryString());
}
/*
SELECT [s].[ID], [s].[Name], [s].[SubID]
FROM [Students] AS [s]
INNER JOIN [Books] AS [b] ON ([s].[ID] = [b].[StudentID]) AND ([s].[SubID] = [b].[SubID])
*/

그런데, "안녕하세요! EntityFramework Linq 질문 드리겠습니다." 질문 글에서는 JOIN에 대해 OR 조건을 추가하고 싶다는 것입니다. 아쉽게도 OR 조건을 직접 명시하는 방법은 없습니다.

하지만 가만 생각해 보면, INNER JOIN의 OR 조건 추가는 사실상 CROSS JOIN과 다를 바가 없습니다. 따라서 다음과 같은 LINQ 쿼리로 구현해도 동일한 resultset을 반환받게 됩니다.

{
    var q = from s in db.Students
            from e in db.Books
            where s.ID == e.StudentID || e.SubID == s.SubID
            select s;

    Console.WriteLine(q.ToQueryString());
}
/* 출력 결과
SELECT [s].[ID], [s].[Name], [s].[SubID]
FROM [Students] AS [s]
CROSS JOIN [Books] AS [b]
WHERE ([s].[ID] = [b].[StudentID]) OR ([b].[SubID] = [s].[SubID])
*/

실제로 위에서 출력한 SELECT 문과 다음의 SELECT 문은 완전히 동일한 결과를 반환합니다.

SELECT [s].[ID], [s].[Name], [s].[SubID]
FROM [Students] AS [s]
INNER JOIN [Books] AS [b] ON ([s].[ID] = [b].[StudentID]) OR ([b].[SubID] = s.SubID)

그나저나, 사실 저도 실무에서 CROSS JOIN을 이제껏 단 한 번도 써본 적이 없습니다. 아마도 LINQ 쿼리의 JOIN에 대한 OR 조건을 검색해 보면 거의 안 나온 이유가 저런 식의 (CROSS JOIN 결과를 낳는) INNER JOIN ... OR을 거의 쓰는 일이 없기 때문일 듯합니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 1/28/2022]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  [127]  128  129  130  131  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
2879정성태3/3/201526646개발 환경 구성: 259. Visual Studio 없이 Visual C++ 컴파일하는 방법
2878정성태2/28/201527458.NET Framework: 503. == 연산자보다는 Equals 메서드의 호출이 더 권장됩니다. [3]파일 다운로드1
2877정성태2/28/201521718.NET Framework: 502. 연산자 재정의(operator overloading)와 메서드 재정의(method overriding)의 다른 점 - 가상 함수 호출 여부 [3]파일 다운로드1
2876정성태2/27/201524205VS.NET IDE: 98. IntegraStudio - Visual Studio에서 Java 프로그램 개발
2875정성태2/26/201522804디버깅 기술: 72. Visual Studio 2013에서의 sos.dll 사용 제한
2874정성태2/26/201519538디버깅 기술: 71. windbg + 닷넷 디버깅 (2) - null 체크 패턴
2873정성태2/25/201537005.NET Framework: 501. FtpWebRequest 타입을 이용해 FTP 파일 업로드 [4]파일 다운로드1
2872정성태2/25/201521152디버깅 기술: 70. windbg + 닷넷 디버깅 (1) - 배열 인덱스 사용 패턴
2871정성태2/24/201525166개발 환경 구성: 258. 윈도우 8.1에서 방화벽과 함께 FTP 서버 여는 (하지만, 권장하지 않는) 방법 [1]
2870정성태2/24/201526266개발 환경 구성: 257. 윈도우 8.1에서 방화벽과 함께 FTP 서버 여는 방법
2869정성태2/23/201520239.NET Framework: 500. struct로 정의한 값 형식(Value Type)의 경우 Equals 재정의를 권장합니다.파일 다운로드1
2868정성태2/23/201524771VS.NET IDE: 97. Visual C++ 프로젝트 디버깅 시에 Step-Into(F11) 동작이 원치 않는 함수로 진입하는 것을 막는 방법 [2]
2867정성태2/23/201518412오류 유형: 273. File History - Failed to initiate user data backup (error 80070005)
2866정성태2/23/201520271오류 유형: 272. WAT080 : Failed to locate the Windows Azure SDK. Please make sure the Windows Azure SDK v2.1 is installed.
1868정성태2/20/201517569오류 유형: 271. The type '...' cannot be used as type parameter 'TContext' in the generic type or method 'System.ServiceModel.DomainServices.EntityFramework.LinqToEntitiesDomainService&lt;T&gt;
1866정성태2/20/201518448오류 유형: 270. "aspnet_regiis -i" 실행 시 0x00000006 오류 해결 방법
1865정성태2/20/201519814.NET Framework: 499. 특정 닷넷 프레임워크 버전 이후부터 제공되는 타입을 사용해야 한다면?
1864정성태2/18/201524806.NET Framework: 498. C#으로 간단하게 만들어 본 ASCII Art 프로그램 [2]파일 다운로드1
1862정성태2/18/201528627.NET Framework: 497. .NET Garbage Collection에 대한 정리 [6]
1861정성태2/18/201523986.NET Framework: 496. 마우스 커서가 놓인 지점의 문자열 얻는 방법 [1]파일 다운로드1
1860정성태2/18/201523822.NET Framework: 495. CorElementType의 요소 값 설명파일 다운로드1
1859정성태2/17/201524196Windows: 106. 컴퓨터를 재부팅하면 절전(Power Saver) 전원 모드로 돌아가는 경우
1858정성태2/16/201534249Windows: 105. 자동으로 로그아웃/잠김 화면 상태로 전환된다면? [2]
1857정성태2/16/201522249.NET Framework: 494. 값(struct) 형식의 제네릭(Generic) 타입이 박싱되는 경우의 메타데이터 토큰 값파일 다운로드1
1856정성태2/15/201521240.NET Framework: 493. TypeRef 메타테이블에 등록되는 타입의 조건파일 다운로드1
1855정성태2/10/201520783개발 환경 구성: 256. WebDAV Redirector - Sysinternals 폴더 연결 시 "The network path was not found" 오류 해결 방법
... 121  122  123  124  125  126  [127]  128  129  130  131  132  133  134  135  ...