简体   繁体   English

Prolog 中的括号抽象

[英]Bracket Abstraction in Prolog

The algorithm "A" according to Antoni Diller looks fairly simple:根据 Antoni Diller 的算法“A”看起来相当简单:

在此处输入图像描述

http://www.cantab.net/users/antoni.diller/brackets/intro.html http://www.cantab.net/users/antoni.diller/brackets/intro.html

Can we do this in Prolog?我们可以在 Prolog 中执行此操作吗?

% associate formulas to left
associate_left(F, A, B) :-
    append(A, [B], F).

% apply algorithm
reduce(b(_, []), []).
reduce(b(A, B), 'K'(B)) :- atom(B), dif(A, B).
reduce(b(A, A), 'I').
reduce(b(A, [F]), R) :- reduce(b(A, F), R). % uncessary paranthesis case
reduce(b(A, F), 'S'(Pr, Qr)) :-
    associate_left(F, P, Q),
    reduce(b(A, P), Pr),
    reduce(b(A, Q), Qr).

I am assuming bound formulas are b(x, F) where x is bound in F.我假设绑定公式是b(x, F) ,其中 x 绑定在 F 中。

?- reduce(b(x, [x]), A).
A = 'I' 

?- reduce(b(x, [y]), A).
A = 'K'(y) 

?- reduce(b(x, [u, v]), A).
A = 'S'('K'(u), 'K'(v)) 

Example from your link您链接中的示例

?- reduce(b(x, [u, v, [w, z, x], [x, z, y], [z, x, [y, x]]]), A).
A = 'S'('S'('S'('S'('K'(u), 'K'(v)), 'S'('S'('K'(w), 'K'(z)), 'I')), 'S'('S'('I', 'K'(z)), 'K'(y))), 'S'('S'('K'(z), 'I'), 'S'('K'(y), 'I'))) 

I tried algorithm B too.我也试过算法B。 It is a bit hairy buthere it is.它有点毛茸茸,但在这里

Used a slightly different coding for application对应用程序使用了稍微不同的编码
with a Prolog operator that is already left associative:使用已保持关联的 Prolog 运算符:

?- X = ((a*b)*c).
X = a*b*c

And added a predicate unlambda/2:并添加了一个谓词 unlambda/2:

unlambda(b(X,Y), Z) :- !,
   unlambda(Y, H),
   reduce(H, X, Z).
unlambda(P*Q, W) :- !,
   unlambda(P, R),
   unlambda(Q, S),
   W = R*S.
unlambda(X, X).

reduce(X, X, W) :- !,
   W = 'I'.
reduce(P*Q, Z, W) :- !,
   reduce(P, Z, R),
   reduce(Q, Z, S),
   W = 'S'*R*S.
reduce(X, _, 'K'*X).

But we see that there is a problem,但是我们看到有一个问题,
results can get quite long:结果可能会很长:

?- unlambda(b(x,b(y,x)), X).
X = 'S'*('K'*'K')*'I'
?- unlambda(b(x,b(y,b(z,(x*z)*(y*z)))), X).
X = 'S'*('S'*('K'*'S')*('S'*('S'*('K'*'S')*('S'*('K'*'K')*('K'*'S')))*
('S'*('S'*('K'*'S')*('S'*('S'*('K'*'S')*('S'*('K'*'K')*('K'*'S')))*
('S'*('S'*('K'*'S')*('S'*('K'*'K')*('K'*'K')))*('S'*('K'*'K')*'I'))))*
('S'*('K'*'K')*('K'*'I')))))*('S'*('S'*('K'*'S')*('S'*('S'*('K'*'S')*
('S'*('K'*'K')*('K'*'S')))*('S'*('S'*('K'*'S')*('S'*('K'*'K')*
('K'*'K')))*('K'*'I'))))*('S'*('K'*'K')*('K'*'I')))

We can use two identities already documented by Curien(*),我们可以使用 Curien(*) 已经记录的两个身份,
that have the effect of [x]E = 'K'E and [x]Ex = E :具有[x]E = 'K'E[x]Ex = E的效果:

reduce(P*Q, Z, W) :- !,
   reduce(P, Z, R),
   reduce(Q, Z, S),
   (S = 'I', R = 'K'*L ->
       W = L;
    S = 'K'*M, R = 'K'*L ->
       W = 'K'*(L*M);
       W = 'S'*R*S).

The result are much better now:结果现在好多了:

?- unlambda(b(x,b(y,x)), X).
X = 'K'
?- unlambda(b(x,b(y,b(z,(x*z)*(y*z)))), X).
X = 'S'

(*) See page 215 rule (abs) and rule (eta): (*) 参见第 215 页规则 (abs) 和规则 (eta):
Categorical Combinators分类组合器
P.-L. P.-L. Curien - 1986居里安 - 1986
https://core.ac.uk/download/pdf/82017242.pdf https://core.ac.uk/download/pdf/82017242.pdf

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM