0

我试图创建一个 Haskell 函数,它将 2 个列表合并到一个列表中,其中新列表中的偶数索引来自列表 1,奇数索引来自列表 2。如果大小不同,则填充 0。

例如:

[1] [10,15,20] => [1,10,0,15,0,20] 
[2,3] [4,5] => [2,4,3,5]

我试图创建几个版本,但没有运气。

我怎么能创造这样的东西?

4

2 回答 2

2

有一个interleave功能,它做类似的事情,但是,不完全是这样。它“合并”列表,直到其中一个结束。

所以你可以自己写这个函数:

merge :: [Int] -> [Int] -> [Int]
merge (x:xs) (y:ys) = x : y : merge xs ys
merge (x:xs) [] = x : 0 : merge xs []
merge [] (y:ys) = 0 : y : merge [] ys
merge _ _ = []

当我们在两边都有一些元素时,我们把它们都拿走。当其中一个元素不存在时,我们取 0 代替它。在所有其他情况下(确实merge [] []如此),我们以递归结束并返回一个空列表。

我们还可以稍微概括我们的函数以支持任何类似数字的类型:

merge :: Num a => [a] -> [a] -> [a]
merge (x:xs) (y:ys) = x : y : merge xs ys
merge (x:xs) [] = x : 0 : merge xs []
merge [] (y:ys) = 0 : y : merge [] ys
merge _ _ = []

此外,我们可以更进一步并使用defData.Default来获取我们类型的默认值,因此我们可以使用这个函数,而不仅仅是数字列表:

import Data.Default

merge :: Default a => [a] -> [a] -> [a]
merge (x:xs) (y:ys) = x : y : merge xs ys
merge (x:xs) [] = x : def : merge xs []
merge [] (y:ys) = def : y : merge [] ys
merge _ _ = []
于 2019-11-25T10:59:05.543 回答
0

使用我的这个答案的想法,使用transpose :: [[a]] -> [[a]]函数,

interweaveWith :: a -> [a] -> [a] -> [a]
interweaveWith def xs ys =
   -- 0 [1] [10,15,20] => [1,10,0,15,0,20] 
   concat $
      zipWith const
         (transpose [ xs ++ repeat def,      -- no limit, padded with def
                      ys ++ repeat def ])
         (transpose [xs, ys])                -- as long as the longest
于 2019-12-03T17:27:15.960 回答