从PowerPack评估合成报价表达式时出错

时间:2010-11-18 08:49:00

标签: f# quotations

我试图找出修改报价然后评估它们。在这里,我开始基本,只是尝试使用Quotations api创建报价。引号绑定好,但在评估时出错。

#r @"FSharpPowerPack-2.0.0.0\bin\FSharp.PowerPack.dll"
#r @"FSharpPowerPack-2.0.0.0\bin\FSharp.PowerPack.Linq.dll"

open Microsoft.FSharp.Quotations
open Microsoft.FSharp.Linq.QuotationEvaluation
open Microsoft.FSharp.Linq

let hardway = 
    Expr.Let(
        new Var("x", typeof<int>),
        Expr.Value(10),
        Expr.GlobalVar("x").Raw)

hardway.EvalUntyped()


Binding session to 'FSharp.PowerPack.Linq.dll'...
System.Collections.Generic.KeyNotFoundException: The given key was not present in the dictionary.
   at Microsoft.FSharp.Collections.MapTreeModule.find[TValue,a](IComparer`1 comparer, TValue k, MapTree`2 m)
   at Microsoft.FSharp.Linq.QuotationEvaluation.ConvExpr(ConvEnv env, FSharpExpr inp) in d:\codeplex\fspowerpack\May2010\src\FSharp.PowerPack.Linq\Linq.fs:line 459
   at Microsoft.FSharp.Linq.QuotationEvaluation.ConvExpr(ConvEnv env, FSharpExpr inp) in d:\codeplex\fspowerpack\May2010\src\FSharp.PowerPack.Linq\Linq.fs:line 704
   at Microsoft.FSharp.Linq.QuotationEvaluation.ConvExpr(ConvEnv env, FSharpExpr inp) in d:\codeplex\fspowerpack\May2010\src\FSharp.PowerPack.Linq\Linq.fs:line 677
   at Microsoft.FSharp.Linq.QuotationEvaluation.CompileImpl[a](a e, Boolean eraseEquality) in d:\codeplex\fspowerpack\May2010\src\FSharp.PowerPack.Linq\Linq.fs:line 837
   at Microsoft.FSharp.Linq.QuotationEvaluation.Expr.EvalUntyped(FSharpExpr ) in d:\codeplex\fspowerpack\May2010\src\FSharp.PowerPack.Linq\Linq.fs:line 854
   at <StartupCode$FSI_0009>.$FSI_0009.main@()
Stopped due to error

3 个答案:

答案 0 :(得分:3)

要使用全局变量来实现这一点,你需要像这样写:

let hardway =  
    Expr.Let( 
        Var.Global("x", typeof<int>), 
        Expr.Value(10), 
        (Expr.GlobalVar<int>("x")) ) 

hardway.EvalUntyped() 

Var.GlobalExpr.Global使用F#引用库使用的一些共享全局变量字典,以便在不显式传递Var值的情况下获取相同的变量实例(如斯金格的解决方案)。

但是,我认为仅创建Var值一次然后保持对对象的引用(并且在表达式中使用相同的对象)会导致代码更易读,所以我更喜欢Stringer的解决方案。 / p>

关于我的代码的几点:

  • 您需要使用Var.Global而不是new Var,因为第二个选项不会将变量存储在全局词典中。
  • 您需要在Expr.GlobalVar中明确指定类型 - 如果您不这样做,F#将使用obj,这是一个不同的变量(它们按名称​​和 type)。

答案 1 :(得分:2)

我不知道如何使用GlobalVar所以我让别人回答这个问题。以下是等待更好解决方案的解决方法:

let hardway = 
    let v = new Var("x", typeof<int>)
    Expr.Let(
        v,
        Expr.Value(10),
        Expr.Var(v))

let res = hardway.EvalUntyped() // res is 10

答案 2 :(得分:0)

Unquote有一个基于自定义反射的评估引擎,它允许您通过传入变量环境来评估合成引用,而不是需要使变量绑定成为表达式本身的一部分。所以你可以做到以下几点:

open Swensen.Unquote
open Microsoft.FSharp.Quotations

let unquoteway = Expr.Var(Var("x", typeof<int>))
let environment = Map.ofList [("x", box 10)]
unquoteway.Eval(environment)

这很有趣,因为您传入的环境是整个表达式评估中用于所有变量绑定和解析的环境,因此可以使用变量范围规则:

let unquoteway = 
    Expr.NewTuple(
       [Expr.Var(new Var("x", typeof<int>))
        Expr.Let(new Var("x", typeof<string>), Expr.Value("hello"), Expr.Var(new Var("x", typeof<string>)))])

let environment = Map.ofList [("x", box 10)]
unquoteway.Eval(environment)

//FSI output:
val unquoteway : Expr = NewTuple (x, Let (x, Value ("hello"), x))
val environment : Map<string,obj> = map [("x", 10)]
val it : obj = (10, "hello")