Expression Trees Spike
// C#, ExpressionTrees
using System;
using System.Linq.Expressions;
namespace Dummy
{
class Program
{
public static void Main(string[] args)
{
//Lambda Expression
Func<int, bool> test = i => i > 5;
Console.WriteLine("Lambda Expression:");
Console.WriteLine(test(2));
Console.WriteLine(test(6));
//C# Compiler (CSC.exe) turns the code above to this: (Microsoft Intermediate Language - MSIL)
Func<int, bool> test2 = GeneratedFuncName;
Console.WriteLine("\nAfter CSC handles:");
Console.WriteLine(test2(2));
Console.WriteLine(test2(6));
//If we wrap it with Expression, CSC does not create a delegate, converts lamda expression to bunch of objects instead.
Expression<Func<int, bool>> exp = i => i > 5;
// *** What CSC did:
ParameterExpression expression3;
Expression<Func<int, bool>> expression =
Expression.Lambda<Func<int, bool>>(
Expression.GreaterThan(expression3 = Expression.Parameter(typeof(int), "i"),
Expression.Constant(5, typeof(int))), new ParameterExpression[] { expression3 });
//Let's look up to the details:
//Expression returns a value. "i > 5" contains 3 expressions, 5, i return the value themselves, and > takes two arguments, returns a bool value.
//Statement returns nothing. Like if, for, while.
//Constant Expressions:
// For the right side of the expression, 5.
ConstantExpression constantExpression = Expression.Constant(5, typeof(int));
Console.WriteLine("\nConstant Expression:");
Console.WriteLine(constantExpression.NodeType);
Console.WriteLine(constantExpression.Type);
Console.WriteLine(constantExpression.Value);
//Parameter Expressions:
//For the left side of the expression, i.
ParameterExpression parameterExpression = Expression.Parameter(typeof(int), "i");
Console.WriteLine("\nParameter Expression:");
Console.WriteLine(parameterExpression.NodeType);
Console.WriteLine(parameterExpression.Type);
Console.WriteLine(parameterExpression.Name);
//Binary Expressions:
//Can be >,<,>= and etc. So in this case, our Binary Expression is GreaterThanExpression.
//For the >.
BinaryExpression binaryExpression = Expression.GreaterThan(parameterExpression, constantExpression);
Console.WriteLine("\nBinary Expression:");
Console.WriteLine(binaryExpression.NodeType);
Console.WriteLine(binaryExpression.Type);
Console.WriteLine(binaryExpression.Left);
Console.WriteLine(binaryExpression.Right);
//Lambda Expression:
//Takes the expression body, which is greater than. And the parameter.
Expression<Func<int, bool>> finalExpression = Expression.Lambda<Func<int, bool>>(binaryExpression, parameterExpression);
Func<int, bool> myDelegate = finalExpression.Compile(); //We directly send our code to JIT (Just in time compiler), and passed CSC.
Console.WriteLine("\nManually created Lambda Expression:");
Console.WriteLine(myDelegate(3));
Console.WriteLine(myDelegate(6));
//IQueryable takes Expression as a paremeter for its Where and Select extensions.
//IEnumerable takes direct delegate.
//This brings us to think that, you can lazy load on IQueryable when getting data from db, viceversa for the IEnumerable.
Console.ReadKey();
}
static bool GeneratedFuncName(int i)
{
return i > 5;
}
}
}