最近,我正在学习F#.我尝试以不同的方式解决问题.像这样:
(*
[0;1;2;3;4;5;6;7;8] -> [(0,1,2);(3,4,5);(6,7,8)]
*)
//head-recursive
let rec toTriplet_v1 list=
match list with
| a::b::c::t -> (a,b,c)::(toTriplet_v1 t)
| _ -> []
//tail-recursive
let toTriplet_v2 list=
let rec loop lst acc=
match lst with
| a::b::c::t -> loop t ((a,b,c)::acc)
| _ -> acc
loop list []
//tail-recursive(???)
let toTriplet_v3 list=
let rec loop lst accfun=
match lst with
| a::b::c::t -> loop t (fun ls -> accfun ((a,b,c)::ls))
| _ -> accfun []
loop list (fun x -> x)
let funs = [toTriplet_v1; toTriplet_v2; toTriplet_v3];
funs |> List.map (fun x -> x [0..8]) |> List.iteri (fun i x -> printfn "V%d : %A" (i+1) x)
Run Code Online (Sandbox Code Playgroud)
我认为V2和V3的结果应该是一样的.但是,我得到以下结果:
V1 : [(0, 1, 2); (3, 4, 5); (6, 7, 8)]
V2 : [(6, 7, 8); (3, 4, 5); (0, 1, 2)]
V3 : [(0, 1, 2); (3, 4, 5); (6, 7, 8)]
Run Code Online (Sandbox Code Playgroud)
为什么V2和V3的结果不同?
Yin*_*Zhu 11
V2使用标准累积变量来完成尾递归:
loop ([0;1;2;3;4;5;6;7;8], []) ->
loop ([3;4;5;6;7;8], [(0,1,2)]) ->
loop ([6;7;8], [(3,4,5), (0,1,2)]) ->
loop ([], [(6,7,8), (3,4,5), (0,1,2)]) ->
[(6,7,8), (3,4,5), (0,1,2)]
Run Code Online (Sandbox Code Playgroud)
V3使用continuation或简单的英语来表示累积函数:
loop ([0;1;2;3;4;5;6;7;8], x->x) ->
loop ([3;4;5;6;7;8], x->(0;1;2)::x) ->
loop ([6;7;8], x->(3;4;5)::x) ->
loop ([], x->(6,7,8)::x) ->
[(6,7,8)] // x->(6,7,8)::x applies to []
->
[(3,4,5);(6,7,8)] // x->(3,4,5)::x applies to [(6,7,8)]
->
[(0,1,2);(3,4,5);(6,7,8)] // x->(0,1,2)::x applies to [(3,4,5);(6,7,8)]
Run Code Online (Sandbox Code Playgroud)
您可以看到累积变量和累积函数之间的区别:
使用累积变量在最后一次调用时停止,因为累积变量存储了答案.但是,累积功能在最后一次调用后仍会进行一些回溯工作.应该注意的是,使用累积函数确实是尾递归的,因为递归调用loop t (fun ls -> accfun ((a,b,c)::ls))实际上是该函数的最后一个语句.
顺便说一句,你提供的代码是一个很好的例子来展示概念尾递归函数.理解这些示例代码的一种方法是像上面两个插图中那样处理小案例.在处理了一些小案例之后,您将更深入地理解这个概念.