SolvedPrivate

LINQ To Entity - Easy Left Join

Posted on 2013-10-24
4
84 Views
Last Modified: 2016-02-10
Hi Experts!

Thanks for reading this.

Trying to convert this T-SQL to Linq to Entity:
SELECT 
  sq.SurveyQuestionText
, sac.SurveyScaleText
FROM [dbo].SurveyQuestions sq
LEFT JOIN 
			(
			SELECT 
			 sa.SurveyQuestionID
			, sc.SurveyScaleText
			FROM SurveyUserAnswers sa
			INNER JOIN SurveyScales as sc
				ON sa.SurveyScaleID = sc.SurveyScaleID
			WHERE sa.UserID = 2
			) As sac
			ON sq.SurveyQuestionID = sac.SurveyQuestionID
WHERE sq.SurveyQuestionCounter <= 10 

Open in new window

This is the INNER JOIN:
SomeDB db = new SomeDB();
var UserAnswered = (from ua in db.SurveyUserAnswers
                    join s in db.SurveyScales on ua.SurveyScaleID equals s.SurveyScaleID
                    where ua.UserID == 2
                    select new 
                    { 
                        ua.SurveyQuestionID,
                        s.SurveyScaleText
                    }).ToList();

Open in new window

A LEFT JOIN looks like this:
var query = (from sq in SurveyQuestions
            from sac in  //UserAnswered above
                             .Where(x => sq.SurveyQuestionID == x.SurveyQuestionID)
                             .DefaultIfEmpty()
            select new
			{sq.SurveyQuestionText
             ,sac.SurveyScaleText}).ToList();

Open in new window


Stuck trying to put the two pieces together.

The query would return All questions and if the user has answered the question it would be null for UserAnswered.

Any ideas? TIA!

If you need more info... This will create the class:
public class SurveyUserAnswer
{
    [Key, Required]
    public int SurveyUserAnswerID { get; set; }
    [Required]
    public int SurveyScaleID { get; set; }
    [Required]
    public int SurveyQuestionID { get; set; }
    [Required]
    public int UserID { get; set; }
    public DateTime DateCreated { get; set; }

    public virtual SurveyScale SurveyScale { get; set; }
    public virtual SurveyQuestion SurveyQuestion { get; set; }
    public virtual User User { get; set; }

    public SurveyUserAnswer()
    {
        DateCreated = DateTime.UtcNow;
    }
}


public class SurveyQuestion
{
    [Key, Required]
    public int SurveyQuestionID { get; set; }
    [Required]
    public int SurveyQuestionCounter { get; set; }
    [Required, MaxLength(500)]
    public string SurveyQuestionText { get; set; }
    public DateTime DateCreated { get; set; }

    public virtual ICollection<SurveyUserAnswer> SurveyUserAnswer { get; set; }

    public SurveyQuestion()
    {
        DateCreated = DateTime.UtcNow;
    }
}

public class SurveyScale
{
    [Key, Required]
    public int SurveyScaleID { get; set; }
    [Required]
    public int SurveyScaleValue { get; set; }
    [Required, MaxLength(8)]
    public string SurveyScaleText { get; set; }
    public DateTime DateCreated { get; set; }

    public virtual ICollection<SurveyUserAnswer> SurveyUserAnswer { get; set; }

    public SurveyScale()
    {
        DateCreated = DateTime.UtcNow;
    }
}

Open in new window

This will create the dummy data content:
        var SurveyScales = new List<SurveyScale> 
        {
        new SurveyScale {SurveyScaleValue = 0, SurveyScaleText = "Never"},
        new SurveyScale {SurveyScaleValue = 1, SurveyScaleText = "Seldom"},
        new SurveyScale {SurveyScaleValue = 2, SurveyScaleText = "Sometime"},
        new SurveyScale {SurveyScaleValue = 3, SurveyScaleText = "Always"}
        };
        SurveyScales.ForEach(r => db.SurveyScales.Add(r));
        db.SaveChanges();

        var SurveyQuestions = new List<SurveyQuestion>
        {
            new SurveyQuestion {SurveyQuestionCounter = 1, SurveyQuestionText = "Question A"},
            new SurveyQuestion {SurveyQuestionCounter = 2, SurveyQuestionText = "Question B"},
            new SurveyQuestion {SurveyQuestionCounter = 3, SurveyQuestionText = "Question C"},
            new SurveyQuestion {SurveyQuestionCounter = 4, SurveyQuestionText = "Question D"}
        };
        SurveyQuestions.ForEach(r => db.SurveyQuestions.Add(r));
        db.SaveChanges();

        var SurveyUserAnswers = new List<SurveyUserAnswer>
        {
            new SurveyUserAnswer {SurveyScaleID = 1,  SurveyQuestionID = 1,  UserID = 1},
            new SurveyUserAnswer {SurveyScaleID = 2,  SurveyQuestionID = 2,  UserID = 1},
            new SurveyUserAnswer {SurveyScaleID = 3,  SurveyQuestionID = 3,  UserID = 1},
            new SurveyUserAnswer {SurveyScaleID = 4,  SurveyQuestionID = 4,  UserID = 1},

            new SurveyUserAnswer {SurveyScaleID = 1,  SurveyQuestionID = 1,  UserID = 2},
            new SurveyUserAnswer {SurveyScaleID = 2,  SurveyQuestionID = 2,  UserID = 2}          
        };
        SurveyUserAnswers.ForEach(r => db.SurveyUserAnswers.Add(r));
        db.SaveChanges();

Open in new window

0
Comment
Question by:allanau20
  • 3
4 Comments
 
LVL 96

Accepted Solution

by:
Bob Learned earned 500 total points
ID: 39599231
You should be able to combine those two queries, if you take advantage of LINQ's laziness.  That means that you need to stay away from things that execute, like ToList.

Example:

var UserAnswered = (from ua in db.SurveyUserAnswers
                    join s in db.SurveyScales on ua.SurveyScaleID equals s.SurveyScaleID
                    where ua.UserID == 2
                    select new 
                    { 
                        ua.SurveyQuestionID,
                        s.SurveyScaleText
                    });

var query = (from sq in SurveyQuestions
            join sac in UserAnswered
                  on sq.SurveyQuestionID equals sac.SurveyQuestionID into leftJoin
            from result in leftJoin.DefaultIfEmpty()
            select new
			{sq.SurveyQuestionText,
                         sac.SurveyScaleText}).ToList();

Open in new window

0
 
LVL 5

Author Comment

by:allanau20
ID: 39599239
Thanks TheLearnedOne; I will try it sometime later today ...
0
 
LVL 5

Author Comment

by:allanau20
ID: 39599879
took your solution and extended it:
var UserAnswered = (from ua in db.SurveyUserAnswers
                    join s in db.SurveyScales on ua.SurveyScaleID equals s.SurveyScaleID
                    where ua.UserID == 2
                    select new
                    {
                        ua.SurveyQuestionID,
                        s.SurveyScaleText,
                        s.SurveyScaleValue
                    });

var query = (from sq in db.SurveyQuestions
                join sac in UserAnswered
                    on sq.SurveyQuestionID equals sac.SurveyQuestionID into leftJoin
                from result in leftJoin.DefaultIfEmpty()
                where sq.SurveyQuestionCounter <= 10 
                select new
                {
                    sq.SurveyQuestionText,  
                    SurveyScaleText = result.SurveyScaleText != null ? result.SurveyScaleText : null,
                    SurveyScaleValue = result.SurveyScaleValue != null ? result.SurveyScaleValue : (int?)null        
                });

Open in new window

0
 
LVL 5

Author Closing Comment

by:allanau20
ID: 39599881
Thanks for your help!!
0

Featured Post

IT, Stop Being Called Into Every Meeting

Highfive is so simple that setting up every meeting room takes just minutes and every employee will be able to start or join a call from any room with ease. Never be called into a meeting just to get it started again. This is how video conferencing should work!

Join & Write a Comment

Suggested Solutions

Today I had a very interesting conundrum that had to get solved quickly. Needless to say, it wasn't resolved quickly because when we needed it we were very rushed, but as soon as the conference call was over and I took a step back I saw the correct …
Exception Handling is in the core of any application that is able to dignify its name. In this article, I'll guide you through the process of writing a DRY (Don't Repeat Yourself) Exception Handling mechanism, using Aspect Oriented Programming.
In this seventh video of the Xpdf series, we discuss and demonstrate the PDFfonts utility, which lists all the fonts used in a PDF file. It does this via a command line interface, making it suitable for use in programs, scripts, batch files — any pl…
This video demonstrates how to create an example email signature rule for a department in a company using CodeTwo Exchange Rules. The signature will be inserted beneath users' latest emails in conversations and will be displayed in users' Sent Items…

760 members asked questions and received personalized solutions in the past 7 days.

Join the community of 500,000 technology professionals and ask your questions.

Join & Ask a Question

Need Help in Real-Time?

Connect with top rated Experts

20 Experts available now in Live!

Get 1:1 Help Now