【问题标题】:F# compiler throws OutOfMemoryExceptionF# 编译器抛出 OutOfMemoryException
【发布时间】:2012-05-16 11:13:57
【问题描述】:

我使用的项目包含许多从单个基类继承的类。在单元测试中,我需要按类型和数据比较接收到的结果。

当我使用 match 在条件列表包含足够多不同条件的情况下按类型进行比较时,编译器会抛出 OutOfMemoryException。

例如,以下 F# 代码在编译期间引发 System.OutOfMemoryException(参数错误 FS0193)(编译大约需要 30 秒才抛出异常)

type IBaseClass() = class end

type IChildClass1 () = inherit IBaseClass () 

type IChildClass2 () = inherit IBaseClass () 

type IChildClass3 () = inherit IBaseClass () 

type IChildClass4 () = inherit IBaseClass () 

type IChildClass5 () = inherit IBaseClass () 

type IChildClass6 () = inherit IBaseClass () 

type IChildClass7 () = inherit IBaseClass () 

type IChildClass8 () = inherit IBaseClass () 

type IChildClass9 () = inherit IBaseClass () 

type IChildClass10 () = inherit IBaseClass () 

type IChildClass11 () = inherit IBaseClass () 

type IChildClass12 () = inherit IBaseClass () 

type IChildClass13 () = inherit IBaseClass () 

type IChildClass14 () = inherit IBaseClass () 

type IChildClass15 () = inherit IBaseClass () 

type IChildClass16 () = inherit IBaseClass () 

type IChildClass17 () = inherit IBaseClass () 

type IChildClass18 () = inherit IBaseClass () 

type IChildClass19 () = inherit IBaseClass () 

type IChildClass20 () = inherit IBaseClass () 


let AreEqual (original: IBaseClass) (compareWith: IBaseClass) : bool =
    match (original, compareWith) with
    | (:? IChildClass1 as a), (:? IChildClass1 as b) -> a = b
    | (:? IChildClass2 as a), (:? IChildClass2 as b) -> a = b
    | (:? IChildClass3 as a), (:? IChildClass3 as b) -> a = b
    | (:? IChildClass4 as a), (:? IChildClass4 as b) -> a = b
    | (:? IChildClass5 as a), (:? IChildClass5 as b) -> a = b
    | (:? IChildClass6 as a), (:? IChildClass6 as b) -> a = b
    | (:? IChildClass7 as a), (:? IChildClass7 as b) -> a = b
    | (:? IChildClass8 as a), (:? IChildClass8 as b) -> a = b
    | (:? IChildClass9 as a), (:? IChildClass9 as b) -> a = b
    | (:? IChildClass10 as a), (:? IChildClass10 as b) -> a = b
    | (:? IChildClass11 as a), (:? IChildClass11 as b) -> a = b
    | (:? IChildClass12 as a), (:? IChildClass12 as b) -> a = b
    | (:? IChildClass13 as a), (:? IChildClass13 as b) -> a = b
    | (:? IChildClass14 as a), (:? IChildClass14 as b) -> a = b
    | (:? IChildClass15 as a), (:? IChildClass15 as b) -> a = b
    | (:? IChildClass16 as a), (:? IChildClass16 as b) -> a = b
    | (:? IChildClass17 as a), (:? IChildClass17 as b) -> a = b
    | (:? IChildClass18 as a), (:? IChildClass18 as b) -> a = b
    | (:? IChildClass19 as a), (:? IChildClass19 as b) -> a = b
    | (:? IChildClass20 as a), (:? IChildClass20 as b) -> a = b
    | _ -> false

当然,我可以将 IEquatable 接口添加到我的 IBaseClass 类中,这将避免使用这种 match 构造,或者将 int Kind 成员(或枚举)添加到 IBaseClass 接口并使 匹配 不是按类型,而是按一些 int 值。

请注意,我尝试在 MS VS 2010 和 MSVS 11 Beta 中编译相同的项目,并且遇到相同的编译器错误

问题:为什么在我的情况下会发生编译器的 OutOfMemoryException(是已知的编译器错误或其他限制),我应该如何重新组织我的 match 条件以避免它?

更新当我将类放入可区分的联合并使用类似的匹配比较时,Fsc.exe 无一例外地编译项目

type AllClasses = 
    | ChildClass1 of IChildClass1 | ChildClass2 of IChildClass2 | ChildClass3 of IChildClass3 | ChildClass4 of IChildClass4 | ChildClass5 of IChildClass5 | ChildClass6 of IChildClass6
    | ChildClass7 of IChildClass7 | ChildClass8 of IChildClass8 | ChildClass9 of IChildClass9 | ChildClass10 of IChildClass10 | ChildClass11 of IChildClass11 | ChildClass12 of IChildClass12
    | ChildClass13 of IChildClass13 | ChildClass14 of IChildClass14 | ChildClass15 of IChildClass15 | ChildClass16 of IChildClass16 | ChildClass17 of IChildClass17 | ChildClass18 of IChildClass18 
    | ChildClass19 of IChildClass19 | ChildClass20 of IChildClass20

let AreEqual2 (original: AllClasses) (compareWith: AllClasses) : bool =
    match (original, compareWith) with
    | ChildClass1(a), ChildClass1(b) -> a = b
    | ChildClass2(a), ChildClass2(b) -> a = b
    | ChildClass3(a), ChildClass3(b) -> a = b
    | ChildClass4(a), ChildClass4(b) -> a = b
    | ChildClass5(a), ChildClass5(b) -> a = b
    | ChildClass6(a), ChildClass6(b) -> a = b
    | ChildClass7(a), ChildClass7(b) -> a = b
    | ChildClass8(a), ChildClass8(b) -> a = b
    | ChildClass9(a), ChildClass9(b) -> a = b
    | ChildClass10(a), ChildClass10(b) -> a = b
    | ChildClass11(a), ChildClass11(b) -> a = b
    | ChildClass12(a), ChildClass12(b) -> a = b
    | ChildClass13(a), ChildClass13(b) -> a = b
    | ChildClass14(a), ChildClass14(b) -> a = b
    | ChildClass15(a), ChildClass15(b) -> a = b
    | ChildClass16(a), ChildClass16(b) -> a = b
    | ChildClass17(a), ChildClass17(b) -> a = b
    | ChildClass18(a), ChildClass18(b) -> a = b
    | ChildClass19(a), ChildClass19(b) -> a = b
    | ChildClass20(a), ChildClass20(b) -> a = b
    | _ -> false

谢谢

【问题讨论】:

  • 我假设您使用IChildClass1-20 仅用于说明目的,或者这是实际代码?我认为编译器本身不应该抛出OutOfMemoryException(但我没有答案)
  • @Abel - 是的,类名仅用于说明目的,类肯定有不同的名称。如果我在 AreEqual 中使用 IChildClass1-18,请注意我的机器 - 项目是可编译的,19 及更多 - 繁荣......异常

标签: f# compiler-errors out-of-memory fsc


【解决方案1】:

这是由 F# 编译器在这种情况下编译元组模式匹配的方式引起的。我不完全确定您何时在这个特定问题中运行以及编译器何时使用其他方法,但这里解释了为什么它在这种情况下会失败......

如果您编写与示例中类似的模式匹配,编译器实质上会生成一个决策树,该决策树测试original (:? IChildClass1) 的第一个模式,然后生成两个分支。第一个分支检查compareWith 是否也是IChildClass1。如果是,则运行第一种情况。然后,其余的模式匹配在两个分支中重复,所以你会得到类似的东西(你可以通过使用ILSpy查看少量情况的编译代码来检查这一点):

if (original is IChildClass1)
  if (compareWith is IChildClass1)
    case #1
  if (original is IChildClass2)
    if (compareWith is IChildClass2)
      case #2
    if (original is IChildClass3)
      (...)
else
  if (original is IChildClass2)
    if (compareWith is IChildClass2)
      case #2
    if (original is IChildClass3)
      (...)

这意味着生成代码的大小与此模式匹配中的案例数量成指数比例。在 20 种情况下,编译器尝试创建 2^20 个分支,但(不出所料)失败了。

【讨论】:

  • 感谢您的解释。是的,我知道 Fsc 创建了 2^20 个分支,这是导致异常的原因。但是,如果我将接口放入可区分的联合中(类型 AllClasses = IChildClass1 的 ChildClass1 | IChildClass2 的 ChildClass2 | IChildClass3 的 ChildClass3 ...)并通过具有可区分联合的元组使用相同的 match - 编译器编译项目时没有问题 - 所以似乎出于某种原因编译后的 if - else 分支用于按类型进行比较和与可区分联合进行比较
  • @Vitaliy 正如我所说 - 我不完全知道编译器如何决定要做什么。但是,编译可区分联合与针对接口进行测试完全不同。对于 DU,编译器知道只有一种情况成立。在接口的情况下,一个值可以匹配多个模式(它可以实现多个接口)。
  • 哈哈。我在 HLVM 中有同样的错误。 flyingfrogblog.blogspot.co.uk/2010/04/…
  • @Tomas - 谢谢你的解释,但修辞问题为什么每个案例只有一个条件是不够的(if (original is IChildClass1 && compareWith is IChildClass1) case #1 ...)
  • @Vitaliy 正如你所说,一个条件就足够了。这只是模式匹配编译器的代码生成部分中的一个错误,它应该考虑子表达式以消除这种膨胀,但作者显然在这种情况下忘记了。轻松完成,轻松修复。
猜你喜欢
  • 1970-01-01
  • 2015-02-28
  • 2014-09-20
  • 1970-01-01
  • 1970-01-01
  • 2012-05-22
  • 2013-03-08
  • 2016-04-01
  • 1970-01-01
相关资源
最近更新 更多