Why can you not use anon function with a dynamic p

2020-05-25 08:35发布

Just ran into this today

An anonymous function or method group cannot be used as a constituent value of a dynamically bound operation.

when trying to do

static R ifNotNull<R>(dynamic o, Func<dynamic, R> returnFunc, R otherwise) {
    return ReferenceEquals(null, o) ? otherwise : returnFunc(o);
}

and use it with

dynamic firstAddress = ...;
return ifNotNull<string>(firstAddress, (a) => a.address_1, null)

Now most of the limitations on dynamics make sense to me - you can't use an extension method because how is the compiler supposed to decide which static to compile it to? But I don't get this here. Where does the confusion come in? What exactly is the limitation?

2条回答
劳资没心,怎么记你
2楼-- · 2020-05-25 09:18

I meant that you need to cast the lambda method to the expect expression you want. Then it'll work just fine.

Like this:

return ifNotNull(firstAddress, (Func<dynamic, string>)((a) => a.address_1), null);
查看更多
够拽才男人
3楼-- · 2020-05-25 09:29

What is the static type of the lamba a => a.address_1? You may be tempted to say it's a Func<dynamic, dynamic>. But remember:

A lambda expression is an anonymous function that you can use to create delegates or expression tree types.

So maybe it's an Expression<Func<dynamic, dynamic>>. A lamda by itself doesn't have a single static type.

Now normally type inference would figure out that you're passing the lamba to a function that takes a Func and it will be converted to a delegate at compile time. However when you are calling with dynamic arguments the method call is dispatched dynamically.

If you have a method call with a dynamic argument, it is dispatched dynamically, period. During the runtime binding, all the static types of your arguments are known (emphasis mine), and types are picked for the dynamic arguments based on their actual values.

So the fact that your method takes a Func isn't take into account, since the actual method call isn't determined until runtime so there is no type inference.

To get this to compile you'll have to cast your lamba to a Func<dynamic, string> as below:

return ifNotNull<string>(firstAddress, new Func<dynamic, string>((a) => a.address_1), null);

Now the static type of your lamda is known.

查看更多
登录 后发表回答