我正在构建一个合并排序函数,我的拆分方法给了我一个值限制错误。我正在使用 2 个累积参数,即拆分产生的 2 个列表,最后我将它们打包成一个元组以供返回。但是我收到值限制错误,我无法弄清楚问题是什么。有人有什么想法吗?
let split lst =
let a = []
let b = []
let ctr = 0
let rec helper (lst,l1,l2,ctr) =
match lst with
| [] -> []
| x::xs -> if ctr%2 = 0 then helper(xs, x::l1, l2, ctr+1)
else
helper(xs, l1, x::l2, ctr+1)
helper (lst, a, b, ctr)
(a,b)
任何意见都值得赞赏。
正如你编写的那样,代码并没有真正的意义。 默认情况下,F# 使用不可变值,因此可以将当前编写的函数简化为:
let split lst =
let a = []
let b = []
(a,b)
这可能不是您想要的。 实际上,由于不可变的绑定,预先声明a, b
和ctr
没有任何价值。
这是一个递归函数,可以解决问题:
let split lst =
let rec helper lst l1 l2 ctr =
match lst with
| [] -> l1, l2 // return accumulated lists
| x::xs ->
if ctr%2 = 0 then
helper xs (x::l1) l2 (ctr+1) // prepend x to list 1 and increment
else
helper xs l1 (x::l2) (ctr+1) // prepend x to list 2 and increment
helper lst [] [] 0
除了使用递归函数,您还可以使用 List.fold
来解决这个问题,fold
是一个高阶函数,它概括了我们在上面的递归函数中明确描述的累积过程。
这种方法更简洁一些,但对于刚接触函数式编程的人来说可能不太熟悉,所以我试图更详细地描述这个过程。
let split2 lst =
/// Take a running total of each list and a index*value and return a new
/// pair of lists with the supplied value prepended to the correct list
let splitFolder (l1, l2) (i, x) =
match i % 2 = 0 with
|true -> x :: l1, l2 // return list 1 with x prepended and list2
|false -> l1, x :: l2 // return list 1 and list 2 with x prepended
lst
|> List.mapi (fun i x -> i, x) // map list of values to list of index*values
|> List.fold (splitFolder) ([],[]) // fold over the list using the splitFolder function