Previously you may have achieved this functionality using datareader. Now that you are using LINQ, you may be wondering how to achieve the same with lists of strongly typed objects.
Quite a few articles out there have reproduced DataReader functionality with LINQ. This article suggests a new approach using the SqlDataSource to create an almost identical code pattern as you previously used with DataReader.
So instead of directly recreating DataReader using LINQ, I’ve replaced it with the same code pattern using a different base class altogether.
old datareader code pattern:
set reader
set connection
set reader parameters
reader loop
get/set reader data
end reader loop
new sqldatasource code pattern:
set sqldatasource
set connection
set sqldatasource parameters
lambda foreach loop calling external delegate function for each item in result
The new approach encapsulates the loop within a LINQ lambda ForEach statement that gets/sets the values and loops through each item inherently. (sourcecode examples below)
Why the hassle?
Sometimes it is useful to load a list of objects from a stored procedure, view or table without immediately binding it to a grid or other data control. The old .NET 2.0 datareader offered a means to easily do this while allowing you the flexibility to do additional work within the reader loop so I sought to reproduce this.
You may be surprised to find the below example does not utilize the LINQ dbml assistance you would normally use in a LINQ to SQL scenario. There is good reason for this as I would like to create a function that allows full control over the list of return values without auto-generating a strange new hybrid return type for every stored procedure.
In addition, I like to fully prototype my applications prior to linking them to the database (that’s how you know you’ve been coding a long time) and this approach makes this much easier.
Simple example:
public class Person { public int Id {get;set;} public string Name {get;set;} public DateTime DateOfBirth {get;set;} public decimal Age {get;set;} //this value isn't actually stored in db and is calculated on load } public static List<Person> LoadPersonsByFilter(string Filter, char Country) { SqlDataSource sds = new SqlDataSource(); sds.ConnectionString = ConfigValues.ConnectionString; //encapsulates ConfigurationManager.ConnectionStrings sds.SelectCommandType = SqlDataSourceCommandType.StoredProcedure; sds.SelectCommand = "get_persons"; sds.SelectParameters.Add("Filter", Filter); sds.SelectParameters.Add("CountryID", Country.ToString()); //sds.Selecting += new SqlDataSourceSelectingEventHandler(sds_Selecting); List<TrainingEvent> results = new List<TrainingEvent>(); sds.Select(new DataSourceSelectArguments()).Cast<DataRowView>().ToList().ForEach(o => LoadPersons_AddResult(o, ref results)); return results; } private static void LoadPersons_AddResult(dynamic o, ref List<Person> results) { results.Add(new Person { Id = Convert.ToInt32(o["PersonID"]), Name = Convert.ToString(o["Name"]), DateOfBirth = Convert.ToDateTime(o["DateOfBirth"]), Age = CalcAge(Convert.ToDateTime(o["DateOfBirth"])) }); } public decimal CalcAge(DateTime DOB) { TimeSpan ts = DateTime.Now-DOB; return Convert.ToDecimal(ts.TotalDays/365); }
At this point, if you are familiar with Linq, you may be thinking of how the same can be accomplished without much effort by simply returning “new” within your linq query. (PM me if you are unsure what I mean by this).
This is true, you could do this, but as complexity increases, you quickly have to look to other alternatives or your LINQ will increase in complexity and create more room for error.
Consider the following strongly typed example:
public List<Computer> Computers {get;set;} //the memory module, cpu and harddrive classes should be self explanatory public class Computer { public int Id {get;set;} public List<MemoryModule> MemoryModules {get;set;} public List<CPU> CPUs {get;set;} public List<HardDrive> HardDrives {get;set;} } public static List<Computers> LoadComputersByFilter(string Filter, char Country) { SqlDataSource sds = new SqlDataSource(); sds.ConnectionString = ConfigValues.ConnectionString; //encapsulates ConfigurationManager.ConnectionStrings sds.SelectCommandType = SqlDataSourceCommandType.StoredProcedure; sds.SelectCommand = "get_computers"; sds.SelectParameters.Add("Filter", Filter); sds.SelectParameters.Add("CountryID", Country.ToString()); //sds.Selecting += new SqlDataSourceSelectingEventHandler(sds_Selecting); List<Computers> results = new List<Computers>(); sds.Select(new DataSourceSelectArguments()).Cast<DataRowView>().ToList().ForEach(o => LoadComputersByFilter_AddResult(o, ref results)); //results might look something like //compid,mmid,cpuid,hdid //1,1,1,1 - denotes first stick //1,2,1,1 - denotes second stick //1,1,1,2 - denotes second hd //1,1,2,1 - denotes second cpu //this is just an example, and would be structured slightly different in a production scenario but the concept remains the same //you can now bind each sub list to its own nested repeater, etc return results; } private static void LoadPersons_AddResult(dynamic o, ref List<Computer> results) { MemoryModule mm = new MemoryModule() { //load info and determine specs or other complex results etc }; CPU cpu = new CPU() { //load info and determine specs or other complex results etc }; HardDrive hd = new HardDrive () { //load info and determine specs or other complex results etc }; results.Add(new Computer { Id=Convert.ToInt32(o["compid"]), MemoryModule = mm, CPU = cpu, HardDrive = hd }); }
As illustrated by the above example, you can easily nest lists within one another without worrying about having to recode a casting mechanism from the custom return type from autogenerated LINQ dbml, or without having to modify the dbml file directly (which resets on updates btw).
The above can be accomplished using LINQ to SQL and lambda expressions, but it will require more practice on your part and is not as explicit IMO.
You also may be wondering why the database call was made by a single stored procedure rather than three separate calls.. If you are unsure, consider the math. Using 1 call to return 4 rows is faster and less work on the database than using 3 calls to return 4 rows.
I should mention there is an alternative approach of creating a custom class that inherits from IEnumerable and can intercept the lazy loading that occurs, but this actually takes more time in my opinion, and may be more prone to error as there are more steps involved.
Enjoy.
meta: Linq DataReader Load Nested List of Objects and Complex Data Results from SqlDataSource
