与此同时,我找到了一个解决方案(经过测试):
这是最初的情况(简化为只有四个功能,但实际上它们更多):
let rec f1 a b c =
f2 a b c;
f3 a b c;
f4 a b c;
and f2 a b c =
f1 a b c;
f3 a b c
f4 a b c
and f3 a b c =
f1 a b c;
f2 a b c
f4 a b c
and f4 a b c =
f1 a b c;
f2 a b c
f3 a b c
这是解决方案:
假设您决定将 f3 移动到另一个文件。然后您可以将上面的文件拆分为两个文件,如下所示:
FILE 1
======
let callRef mf =
match !mf with
| None -> failwith "function ref is none"
| Some f -> f
let r_f3 = ref None;
let rec f1 a1 a2 a3 =
f2 a b c;
callRef r_f3 a1 b1 c1;
f4 a1 b1 c1;
and f2 a1 a2 a3 =
f1 a b c;
callRef r_f3 a1 b1 c1;
f4 a1 b1 c1;
and f4 a1 a2 a3 =
f1 a b c;
f2 a1 b1 c1;
callRef r_f3 a1 b1 c1;
FILE 2
======
let f3 a1 a2 a3 =
f1 a b c;
f2 a1 b1 c1;
f4 an bn cn;
然后,在主初始化函数中(在第三个文件中),你需要做
r_f3 := Some f3;
就是这样。
重复相同的策略将 f1、f2 和 f4 移出第一个文件。
更新:此解决方案适用于返回单位的函数,但不幸的是,对于返回实际类型的函数,它会强制您明确指定函数类型,例如
let (r_f3 : (t1 -> t2 -> t3 -> t4 -> t5) option ref) = ref None;
或者你可以这样做:
let (r_f3 : 'a option ref) = ref None;
但你会得到一个编译器警告。