我正在迈出进入 Haskell 美妙世界的第一步。作为练习,我想实现一个方法来查找列表的最大元素及其索引。我们称这个函数为“maxi”。在列表上调用 maxi 应返回以下结果:
ghci> maxi [1, 3, 4, 1, 2, 3]
(4, 2)
4 是这个列表中最大的 int,它位于索引 2 处。
我试图按如下方式实现此功能:
maxim :: (Ord a) => [a] -> (a, Int)
maxim l =
let pmaxim :: (Ord a) => [a] -> Int -> (a, Int) -- Internal function to do the work
pmaxim [] _ = error "Empty list" -- List is empty, error
pmaxim [x] xi = (x, xi) -- List has one item, return it and the index
pmaxim (x:xs) xi -- More than one item, break list apart
| x > t = (x, xi) -- If current item is bigger, return it and its index
| otherwise = (t, ti) -- If list tail has a bigger item, return that
where (t, ti) = pmaxim xs (ti + 1) -- Get max of tail of the list
in pmaxim l 0 -- Call internal function with start index
当我调用它时,我得到了一些非常奇怪的东西:ghci 在返回最大元素的值后似乎挂起。
ghci> maxi [1, 3, 4, 1, 2, 3]
(4,
我会大胆猜测这与 Haskell 的懒惰评估性质有关,但我发现很难弄清楚这里到底发生了什么,以及如何解决它。我也非常感谢任何人关于如何在 Haskell 中调试的任何提示。有没有一种简单的方法可以在执行期间打印出值而不影响行为?
我只是想指出,我知道有几种更好的方法可以使用内置的 Haskell 函数来获得这种行为。我正在从头开始实施它以尝试学习 Haskell。
谢谢