1

我尝试使用基于 GNU Prolog 的 Prolog 脚本来实现 C 接口。我的问题是获取嵌套 Prolog 列表的单个元素。

实际上我的 C 代码看起来像

...
int func;
PlTerm arg[10];
PlTerm *sol_gb;
PlBool res;
int nmb; 
char *strHead;
char *strTail;
PlLong nummero;
PlTerm pl_nummero;

Pl_Start_Prolog(argc, argv);


Pl_Query_Begin(PL_TRUE);

arg[0] = Pl_Mk_String(strRName);
arg[1] = Pl_Mk_Variable();
arg[2] = Pl_Mk_Variable();
arg[3] = Pl_Mk_String("true");

res = Pl_Query_Call(func, 4, arg);

sol_gb = Pl_Rd_List(arg[2]);
nmb = Pl_List_Length(sol_gb[0]);

strHead = Pl_Write_To_String(sol_gb[0]);      
printf("strHead = %s\n",strHead);
strTail = Pl_Write_To_String(sol_gb[1]);      
printf("strTail = %s\n",strTail);
...

arg[2] 中返回的 Prolog 列表看起来像

[ [ Spezial Bolognese, 
    [2, ,Zwiebeln,300,gramm,Hackfleisch,10, ,Tomaten,
    100,ml,Sahne,500,gramm,Spaghetti] 
  ],
  [ Spaghetti Bolognese,
    [2, ,Zwiebeln gehackt,300,gramm,Hackfleisch,10, ,Fleischtomaten,
     100,ml,Sahne,500,gramm,Spaghetti]
  ]
]

转换成字符串的输出是

strHead = [Spezial Bolognese,[2, ,Zwiebeln gehackt,300,gramm,Hackfleisch,
          10, ,Fleischtomaten,100,ml,Sahne,500,gramm,Spaghetti]]

strTail = [[Spaghetti Bolognese,[2, ,Zwiebeln gehackt,300,gramm,Hackfleisch,
          10, ,Fleischtomaten,100,ml,Sahne,500,gramm,Spaghetti]]]

所以我假设,我“快到了”,但由于我必须重新激活我的 C 知识,我没有得到解决方案如何输入列表的下一级以最终将每个元素作为字符串(“Spezial Bolognese”,下一步:“2”、“Zwiebeln”等)。

如何逐步浏览 C 中的 Prolog 列表?

对于每一个提示,我都会非常高兴,再次感谢您!

4

2 回答 2

2

要从 C 代码中获取列表的内容,您可以使用 2 种函数。

第一种可能性(很简单,因为列表被视为平面对象,但需要更多内存并需要适当的列表,即不适用于未由 [] 终止的列表)

int Pl_Rd_Proper_List_Check(PlTerm the_prolog_list, PlTerm *the_array_receiving_arguments);

这个函数接收一个数组(由你来确保它足够大),将列表的每个元素存储在数组中并返回元素的总数。例子:

PlTerm list = ...some Prolog list...
int nElem = Pl_List_Length(list);
PlTerm *elem = (PlTerm *) calloc(nElem, sizeof(PlTerm));
Pl_Rd_Proper_List_Check(list, elem);
int i;
for(i = 0; i < nElem; i++) { 
   // here there is an argument in elem[i], let's print it
   Pl_Write(elem[i]);
}

第二种可能性(更一般,但将列表视为链表,每个单元格包含头部和尾部(列表))

PlTerm *Pl_Rd_List(PlTerm the_prolog_list);

此函数返回一个包含 2 个元素的数组,对应于接收到的列表的头部和尾部。应该在列表的每个元素上调用此函数;要么你知道元素的数量,要么你测试列表的结尾(例如等待列表 atom [] 的结尾)。这是一个这样做的代码(它应该在上面的循环中,因为我们知道列表的第二个参数是一个嵌套列表。

PlTerm list = ... some Prolog list...;
while(!Pl_Un_Atom(Pl_Atom_Nil(), list)) {
   PlTerm *lst_arg = Pl_Rd_List(list); // [0] = head element, [1] = tail list
   // here there is an argument in lst_arg[0], let's print it
   Pl_Write(lst_arg[0]);
   list = lst_arg[1];
}

在您的示例中,第一个列表如下所示:

[ 'Spezial Bolognese', 
    [2,' ','Zwiebeln',
     300,'gramm','Hackfleisch',
     10,' ','Tomaten',
     100,'ml','Sahne',
     500,'gramm','Spaghetti'] 
]

所以第二个元素是一个嵌套列表。以下代码对上述列表(有 2 个元素)使用第一种方法,对嵌套列表使用第二种方法:

nElem = Pl_List_Length(sol_gb[0]);
PlTerm *elem = (PlTerm *) calloc(nElem, sizeof(PlTerm));
Pl_Rd_Proper_List_Check(sol_gb[0], elem);
int i;
for(i = 0; i < nmb; i++) { 
   if (i != 1) { 
      Pl_Write(elem[i]);
      printf("\n");
   } else {               // we know it is a list
      printf("(");
      PlTerm list = elem[i];
      while(!Pl_Un_Atom(Pl_Atom_Nil(), list)) {
          PlTerm *lst_arg = Pl_Rd_List(list); // [0] = head element, [1] = tail list
          printf(" ");
          Pl_Write(lst_arg[0]);
          list = lst_arg[1];
      }
      printf(" )\n");
   }
}

这里应该是输出

Spezial Bolognese
( 2   Zwiebeln 300 gramm Hackfleisch 10   Tomaten 100 ml Sahne 500 gramm Spaghetti )
于 2014-09-24T15:53:59.573 回答
0

您提供的列表列表的示例代码听起来像是知识表示的一个非常糟糕的选择的教科书示例。我强烈建议您将其更改为更具声明性的表示。就像是:

% pizza(Name, Steps)
pizza('Spezial Bolognese', ...).
...

whereSteps可能是step(...)术语列表。这可能会使处理信息更容易和更有效。例如,在 Prolog 端,您可以使用标准arg/3谓词访问步骤中的特定元素。使用列表,您别无选择,只能在每次需要列表头以外的元素时遍历它们。

于 2014-09-23T11:00:32.053 回答