我尝试使用基于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]
]
]
转换为String的输出是
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”)博洛尼亚“,下一步:”2“,”Zwiebeln“等。)。
如何逐步浏览C中的Prolog列表?
我会对每一个提示感到非常高兴,再次感谢你!
答案 0 :(得分: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 )
答案 1 :(得分:0)
您给出的列表列表的示例代码听起来像是一个教科书示例,它是知识表示的一个非常糟糕的选择。我强烈建议您将其更改为更具声明性的表示形式。类似的东西:
% pizza(Name, Steps)
pizza('Spezial Bolognese', ...).
...
其中Steps
可以是step(...)
条款的列表。这可能使处理信息更容易和更有效。例如,在Prolog端,您可以使用标准arg/3
谓词来访问步骤中的特定元素。对于列表,除了列表头之外的其他元素时,除了遍历它们之外别无选择。