我在 Haskell 中遇到错误,我无法找到

Im having an error with Haskell with cases that i cant find

所以我正在尝试创建这个 AgregarMon 函数,它基本上将 "Monomio" 添加到 "Polinomio" Monomio 最终将成为 Polinomio 中的一个元素,它是一个列表。一点点你就会明白

type Monomio = (Int, Int)
type Polinomio = [Monomio]

agregarMon :: Monomio -> Polinomio -> Polinomio
agregarMon = \m p -> case m of{ (0,0) -> p;
                                 x -> case p of{[] -> [x];
                                                y:ys -> case (snd x == snd y) of { true -> case ((fst x + fst y)==0) of { true -> ys;
                                                                                                                          false -> (fst x + fst y , snd x):ys;}
                                                                                   false -> case snd x < snd y of{true -> y: agregarMon x ys;
                                                                                                                  false -> x:y:ys;}}}}

我已经查看了我的代码一个小时了,但我找不到问题所在。错误说:

Polinomios.hs:45:140: error:
    Unexpected case expression in function application:
        case ((fst x + fst y) == 0) of
          true -> ys
          false -> (fst x + fst y, snd x) : ys
    You could write it with parentheses
    Or perhaps you meant to enable BlockArguments?
   |
45 |                                                                                                 y:ys -> case (snd x == snd y) of { true -> case ((fst x + fst y)==0) of { true -> ys;    |                                                                                                                                            ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^...

第 45 行是上面代码的第四行。对不起,如果我缺少信息,请让我现在。 以防万一,有人不知道,fst和snd在Prelude上。 fst 从 "Monomio" 中获取第一个元素,然后从第二个元素中获取。 fst = (a,b)->a snd (a,b) -> b

虽然您可以在紧要关头使用大括号和分号,但 Haskell 不是基于 C 的语言。 idiomatically uses indentation表示函数作用域

像这样的东西会更地道:

agregarMon :: Monomio -> Polinomio -> Polinomio
agregarMon = \m p -> case m of
                      (0,0) -> p
                      x -> case p of
                            [] -> [x]
                            y:ys -> case (snd x == snd y) of
                                      true -> case ((fst x + fst y)==0) of
                                                true -> ys
                                                false -> (fst x + fst y , snd x):ys
                                      false -> case snd x < snd y of
                                                true -> y: agregarMon x ys
                                                false -> x:y:ys

这可以编译(尽管有警告,见下文),但我不知道它是否按照您的要求执行。 OP 代码不编译,所以很明显 编译的东西不等价。

然而,我所做的只是删除所有花括号和分号,取而代之的是 'fix' 缩进。

在我看来,该代码仍然缩进太多,但至少(有一个例外)它保留了 80 个字符。宽代码可能会迫使人们水平滚动,这不会让你成为很多朋友。

以上代码在 GHCi 中加载,但有警告:

[1 of 1] Compiling Q58986486        ( 58986486.hs, interpreted )

58986486.hs:14:49: warning: [-Woverlapping-patterns]
    Pattern match is redundant
    In a case alternative: false -> ...
   |
14 |                                                 false -> (fst x + fst y , snd x):ys
   |                                                 ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

58986486.hs:15:39: warning: [-Woverlapping-patterns]
    Pattern match is redundant
    In a case alternative: false -> ...
   |
15 |                                       false -> case snd x < snd y of
   |                                       ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^...

58986486.hs:17:49: warning: [-Woverlapping-patterns]
    Pattern match is redundant
    In a case alternative: false -> ...
   |
17 |                                                 false -> x:y:ys
   |                                                 ^^^^^^^^^^^^^^^
Ok, one module loaded.

因此,我不确定它是否按预期工作。

Mark Seemann 的版本可以改进。

首先,让mp成为agregarMon的实参:

agregarMon :: Monomio -> Polinomio -> Polinomio
agregarMon m p = case m of
                      (0,0) -> p
                      x -> case p of
                            [] -> [x]
                            y:ys -> case (snd x == snd y) of
                                      true -> case ((fst x + fst y)==0) of
                                                true -> ys
                                                false -> (fst x + fst y , snd x):ys
                                      false -> case snd x < snd y of
                                                true -> y: agregarMon x ys
                                                false -> x:y:ys

现在我们可以在agruments上使用模式匹配来简化代码:

agregarMon :: Monomio -> Polinomio -> Polinomio
agregarMon (0, 0) p = p
agregarMon x [] = [x]
agregarMon (f, s) (y:ys) = case (snd x == snd y) of
                            true -> case ((fst x + fst y)==0) of
                                         true -> ys
                                         false -> (fst x + fst y , snd x):ys
                            false -> case snd x < snd y of
                                          true -> y: agregarMon x ys
                                          false -> x:y:ys

条件检查结果与 case 模式匹配不是那么惯用,最好使用 if 或守卫:

agregarMon :: Monomio -> Polinomio -> Polinomio
agregarMon (0, 0) p = p
agregarMon x [] = [x]
agregarMon x@(coefx, powx) (y@(coefy, powy):ys)
  | powx == powy = if coefx + coefy == 0
                   then ys
                   else (coefx + coefy, powx):ys
  | powx < powy = y:agregarMon x ys
  | otherwise = x:y:ys

现在我们有一些简单的案例和一个复杂的案例,但即使是复杂的案例也不难阅读和理解(我什至假设您正在使用多项式、系数和幂)。

你可以运行那里:https://repl.it/@Yuri12358/so-monomio-polinomio

也许这个 link 将帮助您提高对在代码中进行分支的不同方法的理解:http://learnyouahaskell.com/syntax-in-functions#pattern-matching

case (snd x == snd y) of { true ->

是错误的:true这里只是一个变量名,与构造函数True无关(大写T!)。因此上面的代码片段等同于

case (snd x == snd y) of { x ->

因此,该模式匹配任何布尔值,TrueFalse。因此,永远不会考虑另一个分支 false -> ...

我建议您打开警告,因为这样做会使 GHC 将第二个分支报告为冗余代码。编译器认为分支无用这一事实表明代码存在严重错误。

最后一点,根据我的经验,代码如下

case something of
  True  -> a
  False -> b

不常见,因为我们可以把它写成

if something
then a
else b

在我看来哪个更容易阅读。