许多谓词本质上使用某种形式的传递闭包,只是发现也必须解决终止问题。为什么不使用以下方法一劳永逸地解决这个问题closure0/3
:
:- meta_predicate closure0(2,?,?).
:- meta_predicate closure(2,?,?).
:- meta_predicate closure0(2,?,?,+). % internal
closure0(R_2, X0,X) :-
closure0(R_2, X0,X, [X0]).
closure(R_2, X0,X) :-
call(R_2, X0,X1),
closure0(R_2, X1,X, [X1,X0]).
closure0(_R_2, X,X, _).
closure0(R_2, X0,X, Xs) :-
call(R_2, X0,X1),
non_member(X1, Xs),
closure0(R_2, X1,X, [X1|Xs]).
non_member(_E, []).
non_member(E, [X|Xs]) :-
dif(E,X),
non_member(E, Xs).
是否存在此定义不能用于实现传递闭包的情况?
为什么要区分 / 2?
详细回答@WouterBeek 的评论:dif/2
或者iso_dif/2
是理想的,因为它们能够显示或发出潜在问题的信号。然而,在当前的实现中,顶层循环通常隐藏了实际问题。考虑一下这个目标closure0(\_^_^true,a,b)
,它本身肯定是有问题的。使用以下系统时,实际问题直接不可见。
| ?- closure0(\_^_^true,a,b). % SICStus
yes
?- closure0(\_^_^true,a,b). % SWI
true ;
true ;
true ...
两个顶级循环都没有显示我们真正想看到的:悬空约束。在 SICStus 中,我们需要一个伪变量来产生一些替换,在 SWI 中,查询必须用call_residue_vars/2
. 以这种方式,现在显示了所有附加了约束的变量。
| ?- closure0(\_^_^true,a,b), Alt=t. % SICStus
Alt = t ? ;
Alt = t,
prolog:dif(_A,a),
prolog:dif(b,_A) ? ;
Alt = t,
prolog:dif(_A,a),
prolog:dif(_B,_A),
prolog:dif(_B,a),
prolog:dif(b,_B),
prolog:dif(b,_A) ...
?- call_residue_vars(closure0(\_^_^true,a,b),Vs). % SWI
Vs = [] ;
Vs = [_G1744, _G1747, _G1750],
dif(_G1744, a),
dif(b, _G1744) ;
Vs = [_G1915, _G1918, _G1921, _G1924, _G1927, _G1930, _G1933],
dif(_G1915, a),
dif(b, _G1915),
dif(_G1921, _G1915),
dif(_G1921, a),
dif(b, _G1921) ...