Solved

Search Filter using lamba or linq - group by & take top record for each group.

Posted on 2016-10-27
3
103 Views
Last Modified: 2016-10-28
Hello,

I am having some difficulties with getting the correct statement in lambda or linq.  


SQL Query
select count(me.schools_id) as numOfStudent, schools_name, members_exams_examdate
from members_exams as me
join schools as s on s.schools_id = me.schools_id
where members_exams_examdate = '8/19/16'
group by s.schools_id, members_exams_examdate, schools_name



This returns
+----------------------+---------------------------+-----------------------------------------+
|numOfStudent |  School Name       |  members exams examdate |
| 1                         | School A                |  8/19/2016                                |
| 4                         | School B                |  8/19/2016                                |
| 7                         | School C                |  8/19/2016                                |
| 21                       | School D                |  8/19/2016                               |
+-----------------------+---------------------------+----------------------------------------+


Controller:   Try1
	public ActionResult Index(string searchBy, string search)

			{
				DateTime convertedSearch;
				DateTime.TryParse(search, out convertedSearch);

					if (searchBy == "members_exams_examdate")
					{

						return View(from x in db.members_exams
									where x.members_exams_examdate == (convertedSearch)
									group x by x.schools_id
										into groups
										select groups.OrderBy(p => p.school.schools_name).First());

					
					}

					return View();
			}

Open in new window

Return Error: The method 'First' can only be used as a final query operation. Consider using the method 'FirstOrDefault' in this instance instead.

Try2:   using  FirstorDefault
select groups.OrderBy(p => p.school.schools_name).FirstorDefault());

Open in new window


Return Error: Error      1      'System.Linq.IOrderedEnumerable<Data.members_exams>' does not contain a definition for 'FirstorDefault' and no extension method 'FirstorDefault' accepting a first argument of type 'System.Linq.IOrderedEnumerable<Data.members_exams>' could be found (are you missing a using directive or an assembly reference?)      


Try3:
	return View(from x in db.members_exams
									where x.members_exams_examdate == (convertedSearch)
									group x by x.schools_id into y
									select new
									{
										schools_id = y.Key
									});

Open in new window


Error: The model item passed into the dictionary is of type 'System.Data.Entity.Infrastructure.DbQuery`1[<>f__AnonymousTypef`1[System.Nullable`1[System.Int16]]]', but this dictionary requires a model item of type 'System.Collections.Generic.IEnumerable`1[Data.members_exams]'.

In the view I would like to group the records and display only the top record for each school.  Just like to table above.


I have tried a number of different things but no success.  

Thanks for the help.
0
Comment
Question by:lovettjay
  • 2
3 Comments
 
LVL 28

Expert Comment

by:Pawan Kumar
ID: 41863238
Can you given me the final output you need , I will write that with SQL Query itself.
0
 
LVL 48

Accepted Solution

by:
PortletPaul earned 500 total points
ID: 41863473
Try using row_number and order that by the count descending
When that column rn is 1 it will be the top row you want
SELECT
      schools_name
    , members_exams_examdate
    , COUNT(me.schools_id)    AS numOfStudent
    , ROW_NUMBER() OVER (ORDER BY COUNT(*) DESC) AS rn
FROM members_exams AS me
JOIN schools AS s ON s.schools_id = me.schools_id
WHERE members_exams_examdate = '20160819'
GROUP BY
      schools_name
    , members_exams_examdate

Open in new window

By the way a date literal of yyyymmdd is THE safest to use in sql server.
Really don't recommend 2 digit years either

{+edit} please note I have changed this (edits via a phone just didn't work)
0
 
LVL 48

Expert Comment

by:PortletPaul
ID: 41863497
by the way...

row_number() over(order by count(*) DESC) as rn

will always result in a single rn = 1 (i.e. one row)
& you might can  include other "tie breaker" column(s) in the ordering

if there is an equal count of students over more than one school, then instead you coud use

rank() over(order by count(*) DESC) as rn

and the rn=1 would apply to more then one school (i.e. rank number of 1)

your call
0

Featured Post

Enterprise Mobility and BYOD For Dummies

Like “For Dummies” books, you can read this in whatever order you choose and learn about mobility and BYOD; and how to put a competitive mobile infrastructure in place. Developed for SMBs and large enterprises alike, you will find helpful use cases, planning, and implementation.

Question has a verified solution.

If you are experiencing a similar issue, please ask a related question

For both online and offline retail, the cross-channel business is the most recent pattern in the B2C trade space.
This article aims to explain the working of CircularLogArchiver. This tool was designed to solve the buildup of log file in cases where systems do not support circular logging or where circular logging is not enabled
Familiarize people with the process of retrieving data from SQL Server using an Access pass-thru query. Microsoft Access is a very powerful client/server development tool. One of the ways that you can retrieve data from a SQL Server is by using a pa…
Via a live example, show how to extract insert data into a SQL Server database table using the Import/Export option and Bulk Insert.

773 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