繁体   English   中英

为什么 Haskell 不接受我的组合“zip”定义?

[英]Why Haskell doesn't accept my combinatoric “zip” definition?

这是教科书的zip功能:

zip :: [a] -> [a] -> [(a,a)]
zip [] _ = []
zip _ [] = []
zip (x:xs) (y:ys) = (x,y) : zip xs ys

我早些时候在#haskell 上问过“zip”是否可以单独使用“foldr”来实现,没有递归,没有模式匹配。 经过一些思考,我们注意到可以使用延续来消除递归:

zip' :: [a] -> [a] -> [(a,a)]
zip' = foldr cons nil
    where
        cons h t (y:ys) = (h,y) : (t ys)
        cons h t []     = []
        nil             = const []

我们还剩下模式匹配。 经过更多的神经元敬酒,我想出了一个我认为合乎逻辑的不完整答案:

zip :: [a] -> [a] -> [a]
zip a b = (zipper a) (zipper b) where
    zipper = foldr (\ x xs cont -> x : cont xs) (const [])

它返回一个平面列表,但会进行压缩。 我确信这是有道理的,但 Haskell 抱怨这种类型。 我继续在一个无类型的 lambda 计算器上测试它,它起作用了。 为什么 Haskell 不能接受我的函数?

错误是:

zip.hs:17:19:
    Occurs check: cannot construct the infinite type:
      t0 ~ (t0 -> [a]) -> [a]
    Expected type: a -> ((t0 -> [a]) -> [a]) -> (t0 -> [a]) -> [a]
      Actual type: a
                   -> ((t0 -> [a]) -> [a]) -> (((t0 -> [a]) -> [a]) -> [a]) -> [a]
    Relevant bindings include
      b ∷ [a] (bound at zip.hs:17:7)
      a ∷ [a] (bound at zip.hs:17:5)
      zip ∷ [a] -> [a] -> [a] (bound at zip.hs:17:1)
    In the first argument of ‘foldr’, namely ‘cons’
    In the expression: ((foldr cons nil a) (foldr cons nil b))

zip.hs:17:38:
    Occurs check: cannot construct the infinite type:
      t0 ~ (t0 -> [a]) -> [a]
    Expected type: a -> (t0 -> [a]) -> t0 -> [a]
      Actual type: a -> (t0 -> [a]) -> ((t0 -> [a]) -> [a]) -> [a]
    Relevant bindings include
      b ∷ [a] (bound at zip.hs:17:7)
      a ∷ [a] (bound at zip.hs:17:5)
      zip ∷ [a] -> [a] -> [a] (bound at zip.hs:17:1)
    In the first argument of ‘foldr’, namely ‘cons’
    In the fourth argument of ‘foldr’, namely ‘(foldr cons nil b)’

至于为什么你的定义不被接受:看看这个:

λ> :t \ x xs cont -> x : cont xs
 ... :: a -> r -> ((r -> [a]) -> [a])

λ> :t foldr
foldr :: (a' -> b' -> b') -> b' -> [a'] -> b'

所以如果你想使用第一个函数作为foldr的参数,你会得到(如果你匹配foldr第一个参数的类型:

a' := a
b' := r
b' := (r -> [a]) -> [a]

这当然是一个问题(因为r(r -> [a]) -> [a]相互递归并且都应该等于b'

这就是编译器告诉你的

如何修复它

您可以使用修复您的想法

newtype Fix a t = Fix { unFix :: Fix a t -> [a] }

它的原始用途中借来的

有了这个,你可以写:

zipCat :: [a] -> [a] -> [a]
zipCat a b = (unFix $ zipper a) (zipper b) where
  zipper = foldr foldF (Fix $ const [])
  foldF x xs = Fix (\ cont -> x : (unFix cont $ xs))

你会得到:

λ> zipCat [1..4] [5..8]
[1,5,2,6,3,7,4,8]

这是(我认为)你想要的。

很明显,您的两个列表都需要属于同一类型,所以我不知道这是否真的对您有帮助

我们可以通过定义一个函数来消除显式模式匹配。

是作弊吗? 如果maybebool被允许,则不会,因为它们是; 那么我们也应该允许list (也在extraData.List.Extra ),

list :: b -> (a -> [a] -> b) -> [a] -> b 
list n c []     = n
list n c (x:xs) = c x xs

一样; 这样我们就可以在您的zip'定义中

cons h t = list [] (\y ys -> (h,y) : t ys)

或例如

         = list [] (uncurry ((:).(h,).fst <*> t.snd))
         = list [] (curry $ uncurry (:) . ((h,) *** t))
         = list [] (flip ((.) . (:) . (h,)) t)
         = list [] ((. t) . (:) . (h,))

如果你喜欢这种东西。

关于你的错误,“infinite type”往往表示自己应用; 事实上,无论你的zipper返回什么,你都是在自我应用它,在你的

zip a b = (zipper a) (zipper b)  where ....

我试图调整你的定义并想出了

zipp :: [a] -> [b] -> [(a,b)]
zipp xs ys = zip1 xs (zip2 ys)
  where
     -- zip1 :: [a] -> tq -> [(a,b)]          -- zip1 xs :: tr ~ tq -> [(a,b)]
     zip1 xs q = foldr (\ x r q -> q x r ) n xs q 
                       -------- c --------
     n    q  = []

     -- zip2 :: [b] -> a -> tr -> [(a,b)]     -- zip2 ys :: tq ~ a -> tr -> [(a,b)]
     zip2 ys x r = foldr (\ y q x r -> (x,y) : r q ) m ys x r  
                         ---------- k --------------
     m  x r  = []

{-
  zipp [x1,x2,x3] [y1,y2,y3,y4]

= c x1 (c x2 (c xn n)) (k y1 (k y2 (k y3 (k y4 m))))
       ---------------       ----------------------
        r                     q

= k y1 (k y2 (k y3 (k y4 m))) x1 (c x2 (c xn n))
       ----------------------    ---------------
        q                         r
-}

它似乎在纸上正确减少,但我仍然在这里遇到了无限的类型错误。

现在没有(立即明显的)自我应用程序,但是第一个 zip 获得的延续类型取决于第一个 zip 本身的类型; 所以仍然存在循环依赖: tqtq ~ a -> tr -> [(a,b)] ~ a -> (tq -> [(a,b)]) -> [(a,b)]

事实上,这是我得到的两种类型错误,(第一个是关于tr类型的),

Occurs check: cannot construct the infinite type:
  t1 ~ (a -> t1 -> [(a, b)]) -> [(a, b)]          -- tr

Occurs check: cannot construct the infinite type:
  t0 ~ a -> (t0 -> [(a, b)]) -> [(a, b)]          -- tq

在使用foldr和 continuations 的通常定义中,这些 continuation 的类型是独立的; 这就是它在那里工作的原因,我猜。

我可以为您提供一个稍微不同的观点(我认为),以得出与 Carsten 类似的解决方案(但类型更简单)。

这是您的代码,用于您的“编织 zip”(我正在为“ r的类型”编写tr ,类似地为“ q的类型”编写tq ;我总是使用“ r ”作为组合函数的递归结果参数foldr定义,作为助记符):

zipw :: [a] -> [a] -> [a]
zipw xs ys = (zipper xs) (zipper ys) where
    zipper xs q = foldr (\ x r q -> x : q r) (const []) xs q
                        --- c -------------- --- n ----

 -- zipper [x1,x2,x3] (zipper ys) =
 -- c x1 (c x2 (c x3 n)) (zipper ys)
         --- r --------  --- q -----  tr ~ tq ; q r :: [a]
                                      --     => r r :: [a]
                                      -- => r :: tr -> [a] 
                                      --   tr ~  tr -> [a]    

所以,这是无限类型。 Haskell 不允许将其用于任意类型(这就是类型变量所代表的含义)。

但是 Haskell 的数据类型确实承认递归。 列表、树等——所有常见的类型都是递归的。 允许的:

data Tree a = Branch (Tree a) (Tree a)

在这里,我们对等式两边相同类型的,就如我们tr的类型等价,两侧tr ~ tr -> [a] 但它是一种特定类型,而不是任意类型。

所以我们只是声明它,遵循上面的“方程”:

newtype TR a = Pack { unpack :: TR a -> [a] } 
           -- unpack :: TR a -> TR a -> [a]

什么是Tree a类型? 它是进入Branch的“东西”,它是一Tree a 给定的树不必无限构造,因为undefined也具有Tree a类型。

什么是TR a类型? 它是进入TR a -> [a]的“东西”,这是一个TR a 给定的TR a不必无限构造,因为const []也可以是TR a类型。

我们想要的递归类型tr ~ tr -> [a]已经成为真正的递归类型定义newtype TR a = Pack { TR a -> [a] } ,隐藏在数据构造函数Pack后面(它将被删除编译器,由于使用了newtype关键字,但这是一个无关紧要的细节;它也适用于data )。

Haskell 在这里为我们处理递归。 类型理论家喜欢自己处理这个问题,用Fix 但是 Haskell 用户已经可以使用该语言。 我们不必了解它是如何实现的,就可以使用它。 无需重新发明轮子,直到我们想自己构建它。

所以, zipper xs类型是tr 现在它变成了TR a ,所以这就是新的zipper xs必须返回的东西——“打包”的列表生成函数。 foldr组合函数必须返回zipper调用返回的内容(根据foldr定义的优点)。 要应用打包函数,我们现在需要先unpack它:

zipw :: [a] -> [a] -> [a]
zipw xs ys = unpack (zipper xs) (zipper ys)
    where
    zipper :: [a] -> TR a
    zipper = foldr (\ x r -> Pack $ \q -> x : unpack q r)
                   (Pack $ const [])

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM