1. 数据结构
到目前为止,所介绍的事实、查询以及规则都使用的是最简单的数据结构。谓词的参数都是原子或者整数,这些都是Prolog的基本组成元素。例如我们所使用过的原子有:
office, apple flashlight, nani
通过把这些最简单的数据组合起来,可以生成复杂的数据类型,我们称之为结构
。结构由结构名
和一定数量的参数组成
。这与以前所学过的目标和事实是一样的。
functor(arg1,arg2,...)
结构的参数可以是简单的数据类型或者是另一个结构。现在在游戏中的物品都是由原子表示的,例如,desk、apple。但是使用结构可以更好的表达这些东西。下面的结构描述了物品的颜色、大小以及重量。
object(candle, red, small, 1).
object(apple, red, small, 1).
object(apple, green, small, 1).
object(table, blue, big, 50).
这些结构可以直接取代原来的location/2
中的参数。但是这里我们再定义一个谓词location_s/2
。注意,虽然定义的结构较为复杂,但是它仍然是location_s/2
的一个参数。
location_s(object(candle, red, small, 1), kitchen).
location_s(object(apple, red, small, 1), kitchen).
location_s(object(apple, green, small, 1), kitchen).
location_s(object(table, blue, big, 50), kitchen).
Prolog的变量是没有数据类型之分的,所以它可以很容易的绑定为结构,如同它绑定为原子一样。事实上,原子就是没有参数的最简单的结构。因此可以有如下的询问。
?- location_s(X, kitchen).
X = object(candle, red, small, 1) ;
X = object(apple, red, small, 1) ;
X = object(apple, green, small, 1) ;
X = object(table, blue, big, 50) ;
no
我们还可以让变量绑定为结构中的某些参数,下面的询问可以找出厨房中所有红色的东西。
?- location_s(object(X, red, S, W), kitchen).
X = candle
S = small
W = 1 ;
X = apple
S = small
W = 1 ;
no
如果不关心大小和重量,可以使用下面的询问,其中变量‘_’
是匿名变量。
?- location_s(object(X, red, _, _), kitchen).
X = candle ;
X = apple ;
no
使用这些结构,可以使得游戏更加真实。例如,我们可以修改以前所编写的can_take/1
谓词,使得只有较轻的物品才能被玩家携带。
can_take_s(Thing) :-
here(Room),
location_s(
object(Thing, _, small,_),
Room
).
同时,也可以把不能拿取某物品的原因说得更详细一些,现在有两个拿不了物品的原因。为了让Prolog在回溯时不把两个原因同时显示出来,我们为每个原因建立一条子句。这里要用到内部谓词not/1
,它的参数是一个目标,如果此目标失败,则它成功;目标成功则它失败。例如,
?- not( room(office) ).
no
?- not( location(cabbage, 'living room') )
yes
- 注意,在Prolog中的
not
的意思是:不能通过当前数据库中的事实和规则推出查询的目标。下面是使用not
重新编写的can_take_s/1
。
can_take_s(Thing) :-
here(Room),
location_s(
object(Thing, _, small, _),
Room
).
can_take_s(Thing) :-
here(Room),
location_s(
object(Thing, _, big, _),
Room
),
write('The '), write(Thing),
write(' is too big to carry.'), nl,
fail.
can_take_s(Thing) :-
here(Room),
not (
location_s(
object(Thing, _, _, _),
Room
)),
write('There is no '), write(Thing), write(' here.'), nl,
fail.
下面来试试功能,假设玩家在厨房里。
?- can_take_s(candle).
yes
?- can_take_s(table).
The table is too big to carry.
no
?- can_take_s(desk).
There is no desk here.
no
原来的list_things/1
谓词也可以加上一些功能,下面的list_things_s/1
不但可以列出房间中的物品,还可以给出它们的描述。
list_things_s(Place) :-
location_s(
object(Thing, Color, Size, Weight),
Place),
write('A '),write(Size),tab(1),
write(Color),tab(1),
write(Thing), write(', weighing '),
write(Weight), write(' pounds'), nl,
fail.
list_things_s(_).
它的回答令人满意多了。
?- list_things_s(kitchen).
A small red candle, weighing 1 pounds
A small red apple, weighing 1 pounds
A small green apple, weighing 1 pounds
A big blue table, weighing 50 pounds
yes
如果你觉得使用1 pounds
不太准确的话,我们可以再使用另一个谓词来解决此问题。
write_weight(1) :-
write('1 pound').
write_weight(W) :-
W > 1, write(W), write(' pounds').
下面试试看
?- write_weight(4).
4 pounds
yes
?- write_weight(1).
1 pound
yes
第一个子句中不需要使用W=1
这样的判断,我们可以直接把1写到谓词的参数中,因为只有为1
时是使用单数,其他情况下都使用复数。第二个子句中需要加入W>1
,要不然当重量为1
时两条子句就同时满足。
结构可以任意的嵌套,下面使用dimension
结构来描述物体的长、宽、高。
object(desk, brown, dimension(6,3,3), 90).
当然,也可以这样来表达物品的特性
object(desk, color(brown), size(large), weight(90))
下面是针对它的一条查询。
location_s(object(X, _, size(large), _), office).
要注意变量的位置哟,不要搞混了。