SWI Prolog 中有类似匿名谓词的东西吗?
Is there something like anonymous predicates in SWI Prolog?
我可以在 SWI Prolog 中定义一个匿名谓词,将它绑定到一个变量,然后再调用它吗?像这样:
?- F = {(X, Y) :- Y is 2 * X}, call(F, 2.0, Y).
就是这样 lambdas
适用于:
?- use_module(library(lambda)).
true.
?- F_2 = (\X^Y^ ( Y is 2*X )), call(F_2,2.0,Y).
F_2 = \X^4.0^(4.0 is 2*X),
Y = 4.0.
或者,在 SWI-Prolog 中,您可以使用 library(yall)。它可以自动加载,所以你不需要导入任何东西:
?- F = [X, Y]>>( Y is 2*X ), call(F, 2.0, Y).
F = [X, 4.0]>>(4.0 is 2*X),
Y = 4.0.
我认为在一般情况下,最好为调用 lambda 的结果使用一个新变量:
?- F = [X, Y]>>( Y is 2*X ), call(F, 2.0, R).
F = [X, Y]>>(Y is 2*X),
R = 4.0.
到目前为止,没有现代 lambda 库支持 Hiord 语法,不要与 Hilog 语法混淆。但是由于在 ISO Prolog 中花括号只是函子 '{}'/1,所以可以用一些规则来做他自己的 Hiord:
'{}'((Formal :- Body), Actual) :-
copy_term(Formal-Body, Actual-Call),
Call.
'{}'((Formal1,Formal2 :- Body), Actual1, Actual2) :-
copy_term(Formal1-Formal2-Body, Actual1-Actual2-Call),
Call.
'{}'((Formal1,Formal2,Formal3 :- Body), Actual1, Actual2, Actual3) :-
copy_term(Formal1-Formal2-Formal3-Body, Actual1-Actual2-Actual3-Call),
Call.
Etc..
以下是一些运行示例:
Jekejeke Prolog 3, Runtime Library 1.3.8 (May 23, 2019)
(c) 1985-2019, XLOG Technologies GmbH, Switzerland
?- F = {X :- write(X), nl}, call(F, hello).
hello
?- F = {X,Y :- Y is X+1}, call(F, 1, R).
R = 2
?- F = {X,Y,Z :- Z is X+Y}, call(F, 1, 2, R).
R = 3
使用 lambda 项时,您会看到
有很多问题。例如,有一个全球性的问题
具有不同解的变量。
除此之外,您可能还对允许柯里化的 lambda 术语感兴趣。 Hiord 术语不允许使用上述实现,例如这里失败:
?- G = {Y :- Y is X+1}, F = {X :- G}, call(F, 1, R).
Error: Undefined or inaccesible predicate {}/1.
{}/1
{}/3
Jekejeke Prologs 你会更幸运
library(experiment/abstract) 也可以进行柯里化:
?- G = Y\(Y is X+1), F = X\G, call(F, 1, R).
R = 2
SWI-Prologs lambda 库也可以进行柯里化:
?- G = [Y]>>(Y is X+1), F = [X]>>G, call(F, 1, R).
R = 2.
而且 Ulrich Neumerkels 库也可以进行柯里化:
?- G = \Y^(Y is X+1), F = \X^G, call(F, 1, R).
R = 2
我可以在 SWI Prolog 中定义一个匿名谓词,将它绑定到一个变量,然后再调用它吗?像这样:
?- F = {(X, Y) :- Y is 2 * X}, call(F, 2.0, Y).
就是这样 lambdas 适用于:
?- use_module(library(lambda)).
true.
?- F_2 = (\X^Y^ ( Y is 2*X )), call(F_2,2.0,Y).
F_2 = \X^4.0^(4.0 is 2*X),
Y = 4.0.
或者,在 SWI-Prolog 中,您可以使用 library(yall)。它可以自动加载,所以你不需要导入任何东西:
?- F = [X, Y]>>( Y is 2*X ), call(F, 2.0, Y).
F = [X, 4.0]>>(4.0 is 2*X),
Y = 4.0.
我认为在一般情况下,最好为调用 lambda 的结果使用一个新变量:
?- F = [X, Y]>>( Y is 2*X ), call(F, 2.0, R).
F = [X, Y]>>(Y is 2*X),
R = 4.0.
到目前为止,没有现代 lambda 库支持 Hiord 语法,不要与 Hilog 语法混淆。但是由于在 ISO Prolog 中花括号只是函子 '{}'/1,所以可以用一些规则来做他自己的 Hiord:
'{}'((Formal :- Body), Actual) :-
copy_term(Formal-Body, Actual-Call),
Call.
'{}'((Formal1,Formal2 :- Body), Actual1, Actual2) :-
copy_term(Formal1-Formal2-Body, Actual1-Actual2-Call),
Call.
'{}'((Formal1,Formal2,Formal3 :- Body), Actual1, Actual2, Actual3) :-
copy_term(Formal1-Formal2-Formal3-Body, Actual1-Actual2-Actual3-Call),
Call.
Etc..
以下是一些运行示例:
Jekejeke Prolog 3, Runtime Library 1.3.8 (May 23, 2019)
(c) 1985-2019, XLOG Technologies GmbH, Switzerland
?- F = {X :- write(X), nl}, call(F, hello).
hello
?- F = {X,Y :- Y is X+1}, call(F, 1, R).
R = 2
?- F = {X,Y,Z :- Z is X+Y}, call(F, 1, 2, R).
R = 3
使用 lambda 项时,您会看到 有很多问题。例如,有一个全球性的问题 具有不同解的变量。
除此之外,您可能还对允许柯里化的 lambda 术语感兴趣。 Hiord 术语不允许使用上述实现,例如这里失败:
?- G = {Y :- Y is X+1}, F = {X :- G}, call(F, 1, R).
Error: Undefined or inaccesible predicate {}/1.
{}/1
{}/3
Jekejeke Prologs 你会更幸运 library(experiment/abstract) 也可以进行柯里化:
?- G = Y\(Y is X+1), F = X\G, call(F, 1, R).
R = 2
SWI-Prologs lambda 库也可以进行柯里化:
?- G = [Y]>>(Y is X+1), F = [X]>>G, call(F, 1, R).
R = 2.
而且 Ulrich Neumerkels 库也可以进行柯里化:
?- G = \Y^(Y is X+1), F = \X^G, call(F, 1, R).
R = 2