Prolog从格式化的文本文件中导入事实

时间:2013-04-13 23:02:00

标签: io prolog

我在文本文件input.txt

中有以下输入
atom1,atom2,atom3
relation(atom1  ,[10,5,2])
relation(atom2  ,[3,10,2])
relation(atom3  ,[6,5,10])

第一行包括文件中关系谓词中使用的原子列表,每个剩余行按第一行列表的顺序表示关系谓词.relation(atom1,[x,y,z])表示atom1有关系第一个原子的值为10,第二个原子的值为5,第三个原子的值为2 我需要读取此文件并单独为每个原子添加表示关系值。例如,这些是将为atom1添加的关系值:

assert(relation(atom1, atom1,10)).
assert(relation(atom1, atom2, 5)).
assert(relation(atom1, atom3, 2)). 

我已经阅读了一些prolog io教程并看到了有关使用DCG的一些建议,但我是一名初学prolog程序员,并且无法选择解决问题的方法。所以我在这里向有经验的prolog程序员寻求帮助。

1 个答案:

答案 0 :(得分:1)

由于您没有说明您正在使用的Prolog,这里有一个用SWI-Prolog编写的代码片段。我尝试通过SWI-Prolog文档参考信号通知 ISO内置。

parse_input :-
    open('input.txt', read, S),
    parse_line(S, atoms(Atoms)),
    repeat,
    (  parse_line(S, a_struct(relation(A, L)))
    -> store(Atoms, A, L), fail
    ;  true ),
    close(S).

:- meta_predicate(parse_line(+, //)).

parse_line(S, Grammar) :-
    % see http://www.swi-prolog.org/pldoc/doc_for?object=read_line_to_codes/2
    read_line_to_codes(S, L),
    L \= end_of_file,
    phrase(Grammar, L).

% match any sequence
% note - clauses order is mandatory
star([]) --> [].
star([C|Cs]) --> [C], star(Cs).

% --- DCGs ---

% comma sep atoms
atoms(R) -->
    star(S),
    (   ",",
        {atom_codes(A, S), R = [A|As]},
        atoms(As)
    ;   {atom_codes(A, S), R = [A]}
    ).

% parse a struct X,
% but it's far easier to use a builtin :)
% see http://www.swi-prolog.org/pldoc/doc_for?object=atom_to_term/3
a_struct(X, Cs, []) :-
    atom_codes(A, Cs),
    atom_to_term(A, X, []).

% storage handler
:- dynamic(relation/3).

store(Atoms, A, L) :-
    nth1(I, L, W),
    nth1(I, Atoms, B),
    assertz(relation(A, B, W)).

使用示例input.txt,我得到了

?- parse_input.
true .

?- listing(relation).
:- dynamic relation/3.

relation(atom1, atom1, 10).
relation(atom1, atom2, 5).
relation(atom1, atom3, 2).
relation(atom2, atom1, 3).
relation(atom2, atom2, 10).
relation(atom2, atom3, 2).
relation(atom3, atom1, 6).
relation(atom3, atom2, 5).
relation(atom3, atom3, 10).

HTH