Why do circular references and recursion make my program fail?

后端 未结 4 1238
耶瑟儿~
耶瑟儿~ 2021-01-14 10:28

I wrote this simple Prolog program.

man(socrates).
mortal(X) :- man(X).
immortal(X) :- immortal(X).

I asked it the usual questions, such as

相关标签:
4条回答
  • 2021-01-14 11:04

    It seems to me that, at least in this case, it would be fairly trivial for Mr. Prolog to conclude that from the rules in the program, it cannot be inferred that Socrates is immortal.

    Prolog uses an incomplete inference algorithm for efficiency. It's meant to be a programming language where programs have a logical meaning in addition to a procedural one, not a full-blown theorem prover. You have to be careful with the order in which you write the clauses, prevent circular definitions, etc.

    As for the logical meaning of your predicate immortal, it's

    immortal(X) -> immortal(X)
    

    which is a tautology and can be removed from your program/theory without changing its logical meaning. This means you should remove it if that helps to improve the procedural meaning (gets rid of an infinite loop).

    0 讨论(0)
  • 2021-01-14 11:16

    Your definitions - and how you interpret them:

    man(socrates).
    

    Socrates is a man.

    mortal(X) :- man(X).
    

    Every man is a mortal.

    immortal(X) :- immortal(X).
    

    Every immortal is immortal.


    Your definitions - and how Prolog interprets them:

    man(socrates).
    

    If you ask about the manhood of Socrates, I know it's true.

    mortal(X) :- man(X).
    

    If you ask me about the mortality of someone, I'll check his manhood (and if that's true, so is the mortality).

    immortal(X) :- immortal(X).
    

    If you ask me about the immortality of someone, I'll check his immortality. (Do you still wonder how that leads to an infinite loop?)


    If you want to state that someone is immortal if he can't be proven mortal, then you can use:

    immortal(X) :- not( mortal(X) ).
    
    0 讨论(0)
  • 2021-01-14 11:23

    Using tabling with XSB:

    :- table foo/1.
    
    foo(X) :- foo(X).
    
    bar(X) :- bar(X).
    

    and then:

    | ?- [tabled].
    [tabled loaded]
    
    yes
    | ?- foo(1).
    
    no
    | ?- bar(1).    % does not finish
    
    0 讨论(0)
  • 2021-01-14 11:26

    How about this little program:

     loopy(Y) :- read(X), Z is X+Y, print(Z), nl, loopy(Y).
    

    Your Mr. Prolog would infer, that loopy(Y) has already been called and would fail.

    0 讨论(0)
提交回复
热议问题