What is the difference between new Action() and a lambda?
Asked Answered
S

3

51

So when I write something like this

Action action = new Action(()=>_myMessage = "hello");

Refactor Pro! Highlights this as a redundant delegate creation and allows me to to shorten it to

Action action = () => _myMessage="hello";

And this usually works great. Usually, but not always. For example, Rhino Mocks has an extension method named Do:

IMethodOptions<T> Do(Delegate action);

Here, passing in the first version works, but the second doesn't. What exactly is going on under the covers here?

Sanative answered 19/4, 2009 at 19:53 Comment(2)
Your second code block does not compile. I get this message "Cannot assign lambda expression to an implicitly-typed local variable". But, if I replace "var" with "Action" it does.Yirinec
Yeah, you're right, it cannot be assigned to an implicitly typed variable, I will edit it.Sanative
S
65

The first version is effectively doing:

Action tmp = () => _myMessage = "hello";
var action = new Action(tmp);

The problem you're running into is that the compiler has to know what kind of delegate (or expression tree) the lambda expression should be converted into. That's why this:

var action = () => _myMessage="hello";

actually doesn't compile - it could be any delegate type with no parameters and either no return value or the same return type as _myMessage (which is presumably string). For instance, all of these are valid:

Action action = () => _myMessage="hello";
Func<string> action = () => _myMessage="hello";
MethodInvoker action = () => _myMessage="hello";
Expression<Action> = () => _myMessage="hello";
// etc

How could the C# compiler work out what type action was meant to be, if it were declared with var?

The simplest way to get round this when calling a method (for your Rhino Mocks example) is to cast:

methodOptions.Do((Action) (() => _myMessage = "hello"));
Smitty answered 19/4, 2009 at 20:2 Comment(5)
VB.Net is able to get around this by generating delegate types on the fly based on the usage. Because VB already differentiates between void and non-void returning functions (sub and function) it makes the differentiation easierTrinatte
"How could the C# compiler work out what type action was meant to be, if it were declared with var?" Simple: function types should be first class structural types, not this named delegate stuff. And quoted code should be noted as such. But I guess that won't change now :).Permutation
I think you need an additional pair of parentheses around the lambda to do such a cast.Awn
@H.B. Done - I'm always amazed at how many brackets you need when casting lambdas :)Smitty
@JonSkeet: That is why i use new Action(() => ...) in those cases.Awn
T
9

Have you verified the second line actually compiles? It should not compile because C# does not support assigning a lambda expression to an implicitly typed variable (CS0815). This line will work in VB.Net though because it supports anonymous delegate creation (starting in VB 9.0).

The Rhino Mocks version does not compile for the same reason the second line should not compile. C# will not automatically infer a type for a lambda expression. Lambda expressions must be used in a context where it is possible to determine the delegate type they are intended to fulfill. The first line works great because the intended type is clear: Action. The Rhino Mocks version does not work because Delegate is more akin to an abstract delegate type. It must be a concrete delegate type such as Action or Func.

For a detailed discussion on this topic, you should read Eric Lippert's blog entries on the subject: http://blogs.msdn.com/ericlippert/archive/2007/01/11/lambda-expressions-vs-anonymous-methods-part-two.aspx

Trinatte answered 19/4, 2009 at 20:0 Comment(0)
F
1

Action is a special type of delegate. So, if you use lambda will not understand which type you want to use since there are many (Action, Func,...).

To overcome this need for cast (which is slow in most of the cases) you can change the parameter of the base function from Delegate to Action:

IMethodOptions<T> Do(Action action);

This way you can use both statements and will not have any different:

Action action = new Action(()=>_myMessage = "hello"); 
Action action = () => _myMessage="hello";

If this is not possible then i suggest to use new Action(() => {}) instead of casting, it would be faster.

Please read the following link for more information on Action and Delegate: https://learn.microsoft.com/en-gb/dotnet/api/system.action?view=netframework-4.7.1#definition

Fougere answered 11/5, 2018 at 9:3 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.