Call prolog predicate from python
I have some .pl
file and I want to call predicate declared in it from py开发者_如何学Gothon script. How can I do that?
For example, test.pl
rD( [], Ans, Ans ).
rD( [X|Xs], Ans, Acc ) :-
member( X, Acc ),
rD( Xs, Ans, Acc ), !.
rD( [X|Xs], Ans, Acc ) :-
\+member( X, Acc ),
append( Acc, [X], AccNew ),
rD( Xs, Ans, AccNew ), !.
Working like
?- rD( [1,2,3,4,5,4], X ).
X = [1, 2, 3, 4, 5].
I want to call rD
somehow from python script and get answer in result variable
result
[1, 2, 3, 4, 5]
ps: it is just an example, and I don't want to rewrite my current Prolog program.
Not that I have a direct experience with it, but there is a project called PySWIP that provides a bridge between Python and SWI-Prolog. The wiki hosted on Google Code's project pages holds installation instructions and some usage examples.
EDIT (5th of July 2019)
PySWIP now seems to be maintained on Github, with its own installation instructions. TLDR: installing SWI-Prolog and pip install pyswip
should do the job, for both Python 2 and 3.
Since you don't want to "rewrite my current Prolog program," I think the natural approach is to make an external call from Python to SWI-Prolog, passing appropriate command line arguments.
Take a look at this SO discussion, How to call external command in Python, from Sept. 2008. Using the subprocess
module allows the stdout from an external command to be piped to the Python process and read as a stream there.
This reduces the problem to choosing command line arguments for SWI-Prolog. It is possible to invoke SWI-Prolog indirectly, through a shell-script on Unix-like systems or a "DOS" batch/cmd file on Windows, but I'll omit further mention of such an indirect call.
See especially the discussion in Sec. 2.4.2 of the SWI-Prolog manual (linked above) of -g
and -t
options. For example:
swipl --quiet -t rD( [1,2,3,4,5,4], X ),halt
will likely do what you want. The --quiet
option suppresses the banner/welcome message, which you probably want to simplify parsing the output as received by Python.
An update for Python3, PySwip in PyPI at the time of writing is for legacy Python only, but the source code on github is Python3 compatible. You can git clone this, run python3 setup.py install
and it'll give you a Python3 version.
To consult an existing knowledge base, stored as knowledge_base.pl:
from pyswip import Prolog
prolog = Prolog()
prolog.consult("knowledge_base.pl")
for res in prolog.query("rD( [1,2,3,4,5,4], X )."):
print(res)
# output:
# {'X': [1, 2, 3, 4, 5]}
from subprocess import Popen, PIPE, STDOUT
p = Popen('/usr/local/sicstus4.2.3/bin/sicstus', shell=True, stdin=PIPE, stdout=PIPE, stderr=STDOUT)
cmd = open('/path/to/your/test.pl').read()
res = p.communicate(cmd)
for line in res:
print line
精彩评论