Entity Framework include with left join is this possible?

后端 未结 3 469
面向向阳花
面向向阳花 2020-12-03 04:43

I have the following tables

  1. ClassRoom (ClassID,ClassName)
  2. StudentClass (StudentID,ClassID)
  3. Student (StudentID,StudentName,Etc..)
相关标签:
3条回答
  • 2020-12-03 04:55

    I just had this problem, in my case it was the EntityTypeConfiguration that was wrong

    I had:

       HasRequired(s => s.ClassRoom)
                    .WithMany()
                    .HasForeignKey(student => student.ClassRoomId);
    

    Instead of:

       HasOptional(s => s.ClassRoom)
                    .WithMany()
                    .HasForeignKey(student => student.ClassRoomId);
    

    It seems HasRequired makes a INNER JOIN while HasOptional makes a LEFT JOIN.

    0 讨论(0)
  • 2020-12-03 05:07

    Exactly:

    1. If StudentDescription.StudentId is nullable -> EF performs a LEFT JOIN, i.e. select * from Student s LEFT JOIN StudentDescription sd on s.StudentID=sd.StudentID.
    2. Otherwise EF does INNER JOIN.
    0 讨论(0)
  • 2020-12-03 05:13

    Yes, it is possible.

    Firstly, .Include does a LEFT OUTER JOIN, using the navigational property you pass through.

    This is how you would explicitly do a LEFT JOIN between Student and StudentDescription:

    var query = from s in ctx.Students
                from sd in s.StudentDescriptions.DefaultIfEmpty()
                select new { StudentName = s.Name, StudentDescription = sd.Description };
    

    As you can see, it's performing the JOIN based on the entity association between Students and StudentDescriptions. In your EF model, you should have a navigational property called StudentDescriptions on your Student entity. The above code is simply using that to perform the join, and defaulting if empty.

    The code is basically identical to .Include.

    Please don't get confused with LEFT JOIN vs LEFT OUTER JOIN.

    They are the same thing.

    The "OUTER" keyword is optional, i believe it is there for ANSI-92 compatability.

    Just .Include everything you need in your query:

    using (var ctx = new TrainingContext())
            {
                studentDo = ctx.Students
                    .Include("ClassRooms")
                    .Include("StudentDescriptions")
                    .Where(x=>x.StudentID==studentId)
                    .Select(x => new StudentDto
                            {
                                StudentId = x.StudentId,
                                StudentName = x.StudentName
                                StudentDescription = x.StudentDescription.Description
                            })
                    .SingleOrDefault();
            }
    

    Basically, make sure all your FK's are expressed as navigational properties on your model, then if so, you don't need to do any joins. Any relationships you require can be done with .Include.

    0 讨论(0)
提交回复
热议问题