我试图创建一个 Haskell 函数,它将 2 个列表合并到一个列表中,其中新列表中的偶数索引来自列表 1,奇数索引来自列表 2。如果大小不同,则填充 0。
例如:
[1] [10,15,20] => [1,10,0,15,0,20]
[2,3] [4,5] => [2,4,3,5]
我试图创建几个版本,但没有运气。
我怎么能创造这样的东西?
我试图创建一个 Haskell 函数,它将 2 个列表合并到一个列表中,其中新列表中的偶数索引来自列表 1,奇数索引来自列表 2。如果大小不同,则填充 0。
例如:
[1] [10,15,20] => [1,10,0,15,0,20]
[2,3] [4,5] => [2,4,3,5]
我试图创建几个版本,但没有运气。
我怎么能创造这样的东西?
有一个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 _ _ = []
此外,我们可以更进一步并使用def
Data.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 _ _ = []
使用我的这个答案的想法,使用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