Haskell:非严格的布尔运算

时间:2012-06-08 15:02:25

标签: haskell lazy-evaluation

无论如何在Haskell中定义如下的函数?

or True      True      = True
or True      undefined = True
or True      False     = True
or undefined True      = True
or undefined False     = undefined
or undefined undefined = undefined
or False     True      = True
or False     undefined = undefined
or False     False     = False

我目前没有它的用​​例(虽然我对它感兴趣),我只是感兴趣,如果可能的话。

2 个答案:

答案 0 :(得分:15)

这在标准的Haskell中是不可能的,但可以通过Conal Elliott在lub库中实现的不安全技巧来完成。

基本上,你写了两个函数:

orL True _ = True
orL False x = x

orR = flip orL

然后您可以将or a b定义为luborL a b的{​​{1}}(相对于“定义”顺序的最小上限)。

在操作上,它并行运行两个计算并选择第一个成功,杀死另一个。

即使按照您的建议运行,它也有重要的缺点。首先,orR a b只有在其论点一致时才是安全的(除非底部相等)。如果你选择lub,结果将是非确定性的,因此违反了纯度!其次,并行运行两个计算的性能开销在某些情况下会变得占主导地位(例如,尝试计算大型列表的lub True False。)。

答案 1 :(得分:12)

根据您的评估顺序(检查值的顺序),您可以编写一个惰性版本:

Prelude> True || undefined
True
Prelude> undefined || True
*** Exception: Prelude.undefined

Prelude> :t or
or :: [Bool] -> Bool

Prelude> or [True, undefined]
True

实际上,Haskell中的默认定义将表现得像这样,因为Haskell是一种惰性语言。

但是,没有办法“跳过”未定义的值,而不先查看值,这会将其评估为底部,这会导致表达式未定义。

请记住,懒惰的值会出现在其中的鬼魂中:

enter image description here

如果你看看盒子内部,鬼可能会找到你。

如果检查底部很重要(例如作为测试套件的一部分),您可以将它们视为例外and intercept them。但你不会在纯粹的功能中这样做。