4

使用 PyClips,我试图在 Clips 中构建规则,从 Python 解释器动态检索数据。为此,我按照手册中的说明注册了一个外部函数。

下面的代码是该问题的一个玩具示例。我这样做是因为我有一个包含大量数据的应用程序,以 SQL 数据库的形式,我想使用 Clips 进行推理。但是,如果我可以简单地将 Clips 直接“插入”到 Python 的命名空间中,我不想浪费时间将所有这些数据转换为 Clips 断言。

但是,当我尝试创建规则时,出现错误。我究竟做错了什么?

import clips

#user = True

#def py_getvar(k):
#    return globals().get(k)
def py_getvar(k):
    return True if globals.get(k) else clips.Symbol('FALSE')

clips.RegisterPythonFunction(py_getvar)

print clips.Eval("(python-call py_getvar user)") # Outputs "nil"

# If globals().get('user') is not None: assert something
clips.BuildRule("user-rule", "(neq (python-call py_getvar user) nil)", "(assert (user-present))", "the user rule")
#clips.BuildRule("user-rule", "(python-call py_getvar user)", "(assert (user-present))", "the user rule")

clips.Run()
clips.PrintFacts()
4

2 回答 2

3

我在 PyClips 支持小组上获得了一些帮助。解决方案是确保您的 Python 函数返回一个 clips.Symbol 对象并使用 (test ...) 来评估 LHS 规则中的函数。似乎还需要使用 Reset() 来激活某些规则。

import clips
clips.Reset()

user = True

def py_getvar(k):
    return (clips.Symbol('TRUE') if globals().get(k) else clips.Symbol('FALSE'))

clips.RegisterPythonFunction(py_getvar)

# if globals().get('user') is not None: assert something
clips.BuildRule("user-rule", "(test (eq (python-call py_getvar user) TRUE))",
                '(assert (user-present))',
                "the user rule")

clips.Run()
clips.PrintFacts()
于 2010-07-28T15:18:29.573 回答
1

您的问题与(neq (python-call py_getvar user) 'None'). 显然剪辑不喜欢嵌套语句。似乎试图将函数调用包装在等式语句中会做坏事。但是,您永远不会断言该值,因为您的函数返回 Nil 或该值。相反,您想要做的是:

def py_getvar(k):
    return clips.Symbol('TRUE') if globals.get(k) else clips.Symbol('FALSE')

然后只需更改"(neq (python-call py_getvar user) 'None')""(python-call py_getvar user)"

这应该有效。在刚刚弄乱它之前没有使用过pyclips,但这应该可以满足您的需求。

>>> import clips
>>> def py_getvar(k):
...     return clips.Symbol('TRUE') if globals.get(k) else clips.Symbol('FALSE')

...
>>> clips.RegisterPythonFunction(py_getvar)
>>> clips.BuildRule("user-rule", "(python-call py_getvar user)", "(assert (user-
present))", "the user rule")
<Rule 'user-rule': defrule object at 0x00A691D0>
>>> clips.Run()
0
>>> clips.PrintFacts()
>>>
于 2010-07-19T20:44:39.970 回答