反映F#Disriminated Union时的奇怪行为

Isa*_*ham 1 reflection f# discriminated-union

以下代码编译并正确运行:

type FooUnion = MyCase of int * string

FSharp.Reflection.FSharpType.GetUnionCases(typeof<FooUnion>)
|> Array.tryFind(fun a -> a.Name = "MyCase")
|> Option.map(fun d -> FSharp.Reflection.FSharpValue.MakeUnion(d, [| 1; "test" |]))
|> Option.bind(function | :? FooUnion as s -> Some s | _ -> None)
Run Code Online (Sandbox Code Playgroud)

但是,如果我删除完全限定的FSharp.Reflection并将其移动到open语句,则代码不再编译: -

open FSharp.Reflection

type FooUnion = MyCase of int * string

FSharpType.GetUnionCases(typeof<FooUnion>)
|> Array.tryFind(fun a -> a.Name = "MyCase")
|> Option.map(fun d -> FSharpValue.MakeUnion(d, [| 1; "test" |]))
|> Option.bind(function | :? FooUnion as s -> Some s | _ -> None)
Run Code Online (Sandbox Code Playgroud)

调用MakeUnion时出错: -

  1. 没有重载符合'MakeUnion'方法[在VS IDE中]
  2. 错误FS0001:这个表达式应该有int类型
    但是这里有类型字符串[如果我手动执行代码,则在FSI内]

thi*_*ing 5

FSharpValue类型包含一个MakeUnion方法:

static member MakeUnion : unionCase:Reflection.UnionCaseInfo * args:obj [] * ?bindingFlags:System.Reflection.BindingFlags -> obj
Run Code Online (Sandbox Code Playgroud)

但FSharp.Reflection命名空间包含具有略微不同签名的扩展方法.

FSharp编译器仅在没有重载时隐式地封装args数组的内容,因此打开命名空间需要将代码更改为:

FSharpValue.MakeUnion(d, [| box 1; box "test" |])
Run Code Online (Sandbox Code Playgroud)

即使您使用完整命名空间前缀.