2014/12/19 Update

改修内容

【制約変数の構造とFrozenGoal惹起タイミングの変更 】

Ver7では変数のユニフィケーションごとにFrozenGoalの惹起を行っていたがVer8においては
ヘッドユニフィケーションが全て成功したあとにFrozenGoalの惹起を行うように変更した。

<Version7.x>

?- freeze(X,write(frozen_goal_invoked)),[1,2,3]=[X,2,4].
frozen_goal_invoked
no

a(1,2).
?- freeze(X,write(frozen_goal_invoked)),a(X,3).
frozen_goal_invoked
no

<Version8.x>

?- freeze(X,write(frozen_goal_invoked)),[1,2,3]=[X,2,4].
no

?- freeze(X,write(frozen_goal_invoked)),a(X,3).
no

 

【虚変数の取り扱い】

プログラム中の虚変数に名前がついていても、コンサルト(アサート)時は虚変数("_")にする。(SWIと同一仕様)これにより、変数名スペルミスを防ぐことが可能。

?-assert(a(A,B)),listing.
a(_,_).
yes

 

【DEBUGGERの大幅改修】

1) SPYの機能拡張
従来の「述語名」、「述語名/アリティ」に加え
「述語名/アリティ/節順位」「述語名/アリティ/節順位/ゴール順位」のSPY(BreakPoint)も可能にした。

| ?-listing.
a(1) :-    true, write(a1), nl.
a(2) :-    true, write(a2), nl.      <== a/1第2節の第2ゴール、write(a2)がTryされるところでトレースを開始したい
a(3) :-    true, write(a3), nl.
yes
| ?-spy a/1/2/2.
yes
||?-a(X),fail.
a1
 [2] 1 Try   : write(a2) ? w         <== "w"コマンドで現在位置確認
         a/1/2/2                     <== トレース開始点が a/1/2/2であることが確認できる
 [2] 1 Try   : write(a2) ? p         <== "p"コマンドで呼び出し祖先を確認  
         1:  write(a2)
         0:  a(2) :-
                true,
             *> write(a2),           <== 親節のこの位置(a/1/2/1)をTry中
                nl.
         Goal:  ?- a(2),fail.
 [2] 1 Try   : write(a2) ?

2) Trace コマンドの追加
トレース中のコマンドに次の2つを追加。

(1)w(where) コマンド Trace中のゴールの位置情報(述語名/アリティ/節順位/ゴール順位)を表示

| a:- b.        % a/0/1  述語名=a,アリティ=0 ,第1節
| a:- c.        % a/0/2  述語名=a,アリティ=0 ,第2節

||?-a.
 [1] 0 Try   :a ?  
  Match   : a :-
                b. ? w  <== wコマンド
          a/0/1         <== a/0 の第1節にマッチしました
 [2] 1 Try   :b ? w     <== wコマンド
          a/0/1/1       <== a/0 の第1節の1番目のゴール(b)をTryしています

(2)i(inspect) コマンド 制約変数の制約情報調査機能

||?-X in 1..10,X in 5..20.
 [1] 0 Try   : X_17 in 1..10,X_17 in 5..20 ? 
 << BUILTIN CALL >>
 [1] 0 Try   : X_17 in 1..10 ? 
 << BUILTIN CALL >>
 [1] 0 Succ  : _27 in 1..10 ? i    <== "i" コマンド

        _27 { [1..10]::true }      <== 変数が 1..10 の値領域に束縛されたのが確認できる
         in 1..10
 [1] 0 Succ  : _27 in 1..10 ? 
 [1] 0 Try   : _27 in 5..20 ? 
 << BUILTIN CALL >>
 [1] 0 Succ  : _27 in 5..20 ? i    <== "i" コマンド

        _27 { [5..10]::true }      <== 5..20 により、変数が 5..10 の値領域になったのが確認できる
         in 5..20
 [1] 0 Succ  : _27 in 5..20 ?

3)Parent表示の改良
(1) スタック深さ番号を正規化(従来と逆順)下記の 3: 2: 1:
(2) 祖先節の表示でTry中のゴールに"*>"を追加

[4] 3 Try   : select([4,3,2,1],S_70,D_72) ? p   <== "p"コマンドで先祖である節をすべて表示します
         3:  select([4,3,2,1],S_70,D_72)
         2:  put([4,3,2,1],[],L_8) :-           <== select/3の親はput/3です
             *> select([4,3,2,1],S_70,D_72),    <== ここをTry中
                safe([],S_70,S_70),
                put(D_72,[S_70],L_8).
         1:  queen(4,L_8) :-                    <== put/3の親はqueen/2です
                generate(4,[4,3,2,1]),
             *> put([4,3,2,1],[],L_8).          <== ここをTry中
         0:  q(4) :-                            <== queen/2の親はq/1です
                e_register(0,0,0),
             *> queen(4,L_8),                   <== ここをTry中
                e_register(0,N1_10,N1_10+1),
                M_12 is N1_10+1,
                write(No.),
                write(M_12),
                nl,
                disp(L_8,4),
                fail.
         Goal:  ?- q(4).                        <== q/1を呼び出したトップレベルのゴールです
 [4] 3 Try   : select([4,3,2,1],S_70,D_72) ?

4)leash/1,leash2/1 の改良
従来のビット対応設定に加え、局面頭文字による設定を追加。

?- leash1(rfpt).  <== retry,fail,pop,try の意味 2'111001 と同義

5)leash2(2) モードでのトレース表現の改良
各局面表示の直後にWコマンドの内容を表示。

6)Trace中のBreakの振る舞い
従来、SPY,Callcountのクリアをしていたが、これを廃止。

7) Trace中の項の表示をwriteqに変更

8) Trace コマンドに d (Dump Stack)を追加

D
dump stack
スタックの最底部から最上部までのゴールを全て木構造で表示します。
”[シリアルナンバー] スタックの深さ: ゴール” の形式で表示され、”>*”が現在のスタック位置を表します。
[シリアルナンバー] はトレース時の「インタプリテーションが始まってからの実際のゴールの積層数」と同じです。
”スタックの深さ:”は逆順ですが、Pコマンドと同じ現在ゴールの直系列で「現在ゴールからトップゴールまでの先祖系列の深さ」です。
”スタックの深さ:”がついていない行はオルタナティブの残っている兄弟ゴールおよびその子孫にあたります。
[シリアルナンバー] はバックトラックパス(番号の小さい方向に向かって順番にバックトラックする)でもあります。

<例>

[14] 1 Try   : e_register(0,N1_10,N1_10+1) ? d     <== Dコマンド
     [1]   0:  q(4)
     [2]       ┣ queen(4,[2,4,1,3])
     [3]       ┃ ┗ put([4,3,2,1],[],[2,4,1,3])
     [4]       ┃   ┣ select([4,3,2,1],3,[4,2,1])
     [5]       ┃   ┃ ┗ select([3,2,1],3,[2,1])
     [6]       ┃   ┗ put([4,2,1],[3],[2,4,1,3])
     [7]       ┃     ┣ select([4,2,1],1,[4,2])
     [8]       ┃     ┃ ┗ select([2,1],1,[2])
     [9]       ┃     ┃   ┗ select([1],1,[])
    [10]       ┃     ┗ put([4,2],[1,3],[2,4,1,3])
    [11]       ┃       ┣ select([4,2],4,[2])
    [12]       ┃       ┗ put([2],[4,1,3],[2,4,1,3])
    [13]       ┃         ┗ select([2],2,[])
    [14]>* 1:  ┗ e_register(0,N1_10,N1_10+1)    Try
 [14] 1 Try   : e_register(0,N1_10,N1_10+1) ? 

【TopLevelの結果表示をwriteqに変更】

<例>

Version 8.08まで

| a('X').
| ?- a(X).
X	= X

Version 8.09

| ?- a(X).
X	= 'X'