我目前正在学习 Haskell。我选择这种语言的动机之一是编写具有高度鲁棒性的软件,即完全定义的、数学上确定的、永远不会崩溃或产生错误的函数。我并不是指由系统谓词(“系统内存不足”、“计算机着火”等)引起的故障,这些并不有趣,并且可以简单地使整个过程崩溃。我也不是指由无效声明(pi = 4
)引起的错误行为。
相反,我指的是由错误状态引起的错误,我想通过严格的静态类型使这些状态不可表示和不可编译(在某些函数中)来消除这些错误。在我看来,我称这些函数为“纯粹的”,并认为强类型系统可以让我完成这个任务。然而,Haskell 并没有以这种方式定义“纯”,并允许程序error
在任何上下文中崩溃。
这是完全可以接受的,一点也不奇怪。然而令人失望的是,Haskell 似乎没有提供一些功能来禁止可能导致分支使用error
.
这是一个人为的例子,为什么我觉得这令人失望:
module Main where
import Data.Maybe
data Fruit = Apple | Banana | Orange Int | Peach
deriving(Show)
readFruit :: String -> Maybe Fruit
readFruit x =
case x of
"apple" -> Just Apple
"banana" -> Just Banana
"orange" -> Just (Orange 4)
"peach" -> Just Peach
_ -> Nothing
showFruit :: Fruit -> String
showFruit Apple = "An apple"
showFruit Banana = "A Banana"
showFruit (Orange x) = show x ++ " oranges"
printFruit :: Maybe Fruit -> String
printFruit x = showFruit $ fromJust x
main :: IO ()
main = do
line <- getLine
let fruit = readFruit line
putStrLn $ printFruit fruit
main
假设我偏执于纯函数readFruit
并且printFruit
真的不会由于非手动状态而失败。你可以想象代码是用于发射一个装满宇航员的火箭,在一个绝对关键的例程中需要序列化和反序列化水果值。
第一个危险自然是我们在模式匹配中犯了一个错误,因为这给了我们无法处理的可怕错误状态。值得庆幸的是,Haskell 提供了内置的方法来防止这些,我们只需编译 -Wall
包含-fwarn-incomplete-patterns
和 AHA 的程序:
src/Main.hs:17:1: Warning:
Pattern match(es) are non-exhaustive
In an equation for ‘showFruit’: Patterns not matched: Peach
我们忘记序列化 Peach fruits 并且showFruit
会抛出一个错误。这很容易解决,我们只需添加:
showFruit Peach = "A peach"
该程序现在可以在没有警告的情况下编译,避免了危险!我们发射了火箭,但突然程序崩溃了:
Maybe.fromJust: Nothing
由于以下故障线路,火箭注定要坠入大海:
printFruit x = showFruit $ fromJust x
本质fromJust
上它有一个分支,Error
所以如果我们尝试使用它,我们甚至不希望程序编译,因为printFruit
绝对必须是“超级”纯的。我们可以解决这个问题,例如将行替换为:
printFruit x = maybe "Unknown fruit!" (\y -> showFruit y) x
我觉得很奇怪 Haskell 决定实现严格的类型化和不完整的模式检测,这一切都是为了防止无效状态被表示,但却因为没有给程序员一种方法来检测分支error
何时出现而落到了终点线的前面不允许。从某种意义上说,这使得 Haskell 不如 Java 健壮,后者迫使您声明允许您的函数引发的异常。
实现这一点的最简单的方法error
是以某种方式简单地 undefined,通过某种形式的关联声明,在本地为函数及其方程使用的任何函数。然而,这似乎是不可能的。
关于错误与异常的 wiki 页面为此通过合同提到了一个名为“扩展静态检查”的扩展,但它只会导致链接断开。
它基本上归结为:我如何让上面的程序因为它使用而无法编译fromJust
?欢迎所有想法、建议和解决方案。