Suppose you want to create a List < string> of names of files residing in a directory. This is how you could do it.
Set up an array of FileInfo objects:
string filePath = @"C:\Temp";
DirectoryInfo dInfo = new DirectoryInfo(filePath);
FileInfo[] fileInfoArray = dInfo.GetFiles();
Create a string type List:
List<string> fileList = new List<string>();
foreach (FileInfo fi in fileInfoArray)
{
fileList.Add(fi.Name);
}
Select
Using lambda expressions and the Select extension method it is possible to re-write the last 3 lines as follows:
var files = fileInfoArray.Select(fi => fi.Name);
That's one line instead of 3. But we still want a List <string> object to be used later in the code. We need to use the Enumerable.ToList <TSource> method to cast to a List. This is how we do it:
List<string> fileList2 = fileInfoArray.Select(fi => fi.Name).ToList ();
This way you use one line of code instead of 4.
Modify Elements in a Collection
Use the Select method to modify elements in a collection. The important part is you need to return each element;
An example follows:
var plans = query.AsEnumerable().Select(it =>
{
it.PlanYears = Enumerable.Range(it.Plan.StartDate.Year, (DateTime.Now.Year - it.Plan.StartDate.Year) + 1);
return it;
});
How to get an ordinal position of an item in the list.
Example:
var mm = new System.Globalization.DateTimeFormatInfo().MonthNames.Select( (a, i) => new { Item = a, Position = i });
Where
Now, what if we add only those file names to the List that meet certain criteria, e.g:
foreach (FileInfo fi in fileInfoArray)
{
if (fi.Length > 10000000)
{
largeFiles.Add(fi.Name);
}
}
Which equivalent extension method can we use to achieve the same result? We need to use the Where extension method of the IEnumerable <T> interface first, and then append the earlier Select method to the statement:
var varLargeFiles = fileInfoArray.Where (fi => fi.Length > 1000000).Select(fi=>fi.Name);
FindAll
If you need to filter your collection and iterate through the result, use the FindAll extension:IncomeSourceList.FindAll(item => item.ItemSourceType == SourceType.PERM).ForEach(
item => Debug.WriteLine(item.ItemSourceType)
);
This code selects only those collection items whose ItemSourceType equals to PERM type, and then loops through the result.
LINQ
Now, to complete the example, I'll show the way to obtain a similar list using the LINQ:
var q = from fi in fileInfoArray
where fi.Length > 1000000
select fi.Name;
List<string> largeFiles2 = q.ToList();
Query operator ToList() forces immediate query evaluation.
Boolean Methods
What if you have a method that loops through an array and returns a bool type depending on some condition inside the loop. Can we use query operators instead? It is slightly more difficult but still possible.
Suppose you have the following methods that compares each character in a string array to each character in another string array and returns false when a character in the second array is not found in the first array:
public static bool IsAnagram(string word, string input)
{
char[] inputArray = input.ToCharArray();
foreach (char ch in word.ToCharArray())
{
if (!inputArray.Contains(ch))
return false;
}
return true;
}
To emulate similar statements, first we need to find the first non-matching character:
char c = word.ToCharArray().FirstOrDefault (x => !inputArray.Contains(x));
Now we can return a bool value depending on the result of the previous statement:
return (c != 0) ? false: true;
For sheer fun you can rewrite those two statements as follows:
return (word.ToCharArray().FirstOrDefault(x => !inputArray.Contains(x))) == 0;
Grouping
Grouping allows us to group the result into distinct groups using a field or several fields as group criteria. Each group will have a key property that you can use to refer to the field(s) grouped on.Example:
Let's say we have a Sales table with the following fields: Rep, SalesDate, RepEmail, etc. We want to know how many sales each Rep made each month.
First, let's build a query that counts the total number of sales per Rep:
from s in Sales group s by s.Rep into gr select new { Name = gr.Key, Count = gr.Count() }
This query would translate into the following sql statement:
select Rep as Name, Count(*) from Sales Group By Rep
To group the result further by month, we would use the following query:
from s in Sales group s by new {s.Rep, s.SalesDate.Month} into gr select new { Name = gr.Key.Rep, SalesMonth = gr.Key.Month, Count = gr.Count() }
Sorting
This does not really belong in this post, however, here is a quick way to sort a list of objects. Your object could be similar to this one:
class MyObject{ public string Symbol {get;set;} public double Value {get;set;} } myList.Sort((a, b) => a.Symbol.CompareTo(b.Symbol));
Any
To find out if an element exists in a collection, use the extension method Any:
if(userRoles.Any(role=> role.Equals("Admin")))
...
Some other posts you might find interesting:
Query Operator First
Easy Syntax to Print List Elements