这里有什么可以在C#中完成 -
var two = 2;
System.Linq.Expressions.Expression<System.Func<int, int>> expr = x => x * two;
expr.Compile().Invoke(4); // returns 8
我希望在F#中做到精确的等价物。这是我尝试过的,但没有编译 -
let two = 2
let expr = (fun x -> x * two) : System.Linq.Expressions.Expression<System.Func<int, int>>
expr.Compile().Invoke(4) // desired to return 8
也许可以预见,编译在第2行失败并出现以下错误 -
"This function takes too many arguments, or is used in a context where a function is not expected."
let expr = (fun x -> x * two) : System.Linq.Expressions.Expression<System.Func<int, int>>
^^^^^^^^^^^^^^^^
答案 0 :(得分:19)
我不确定为什么要避免使用F#引用 - 在封面下,它们与C#表达式树基本相同,如果要在F#中创建表达式树,编译器将使用引号无论如何......在封面下......
无论如何,您可以在不编写显式<@ .. @>
的情况下执行此操作,因为编译器可以在将函数作为参数传递给方法时自动引用该函数。所以你可以这样做:
type Expr =
static member Quote(e:Expression<System.Func<int, int>>) = e
let two = 2
let expr = Expr.Quote(fun x -> x * two)
expr.Compile().Invoke(4) // desired to return 8
编辑:但是,这实际上编译为包含在将其转换为C#表达式树的调用中的F#引用。所以,最后,你会得到和你写的一样:
open Microsoft.FSharp.Linq.RuntimeHelpers
let two = 2
let expr =
<@ System.Func<_, _>(fun x -> x * two) @>
|> LeafExpressionConverter.QuotationToExpression
|> unbox<Expression<Func<int, int>>>
expr.Compile().Invoke(4) // desired to return 8