What does Expression.Reduce() do?
With a little disassembling, I found that Expression.CanReduce always reutrns false
and Expression.Reduce() always returns this
. However, there are a few types that override both. LambdaExpression inherits the default implementations, which explains why the expressions that have been tried so far do not work.
One of the types that overrides Reduce() is MemberInitExpression, which led me to the following successful experiment:
class ReduceFinder : ExpressionVisitor {
public override Expression Visit(Expression node) {
if (node != null && node.CanReduce) {
var reduced = node.Reduce();
Console.WriteLine("Found expression to reduce!");
Console.WriteLine("Before: {0}: {1}", node.GetType().Name, node);
Console.WriteLine("After: {0}: {1}", reduced.GetType().Name, reduced);
}
return base.Visit(node);
}
}
class Foo {
public int x;
public int y;
}
static class Program {
static void Main() {
Expression<Func<int, Foo>> expr = z => new Foo { x = (z + 1), y = (z + 1) };
new ReduceFinder().Visit(expr);
}
}
Output:
Found expression to reduce!
Before: MemberInitExpression: new Foo() {x = (z + 1), y = (z + 1)}
After: ScopeN: { ... }
This is a fairly old question but it seems to have a bit of interest, so I'm adding this extra response with information on what the out-of-box .NET stuff does as of right now.
As far as I can tell, Reduce() is only overridden in complex operations that implement an assignment as part of their work. There seem to be three key scenarios.
Compound assignments are expanded to discrete binary arithmetic and assignment operations; in other words,
x += y
becomes
x = x + y
.Pre-increment and post-increment operators are expanded to their discrete operations. For pre-increment/decrements,
++x
becomes approximately:
x = x + 1
and
x++
becomes approximately:
temp = x; x = x + 1; temp;
I say approximately because the operation is not implemented as a binary operation
x + 1
with the left operand beingx
and the right operand being the constant1
but as a unary increment/decrement operation. The net effect is the same.Member and list initializers are expanded from their short form to their long form. So:
new Thing() { Param1 = 4, Param2 = 5 }
becomes:
temp = new Thing(); temp.Param1 = 4; temp.Param2 = 5; temp;
and:
new List<int>() { 4, 5 }
becomes:
temp = new List<int>(); temp.Add(4); temp.Add(5); temp;
Whether these changes make it easier or harder for a person to implement something that parses an expression tree is a matter of opinion, but bottom line is that's the level of reduction that seems to come out-of-box in the .NET framework.
The document you need to look at is expr-tree-spec.pdf.
This is the specification for the expression trees. Read the "2.2 Reducible Nodes" and "4.3.5 Reduce Method" sections.
Basically, this method is intended for people implementing or porting their dynamic langauges to .NET. So that they can create their own nodes that can "reduce" to standard expression tree nodes and can be compiled. There are some "reducible" nodes in the expression trees API, but I don't know whether you can get any practical examples (since all standard expression nodes compile anyway, as the end-user you probably do not care whether they are "reduced" behind the scenes or not).
Yes, MSDN documentation is very basic in this area, because the main source of info and docs for language implementers is on GitHub, with the documentation in its own subfolder.