【发布时间】:2016-12-28 10:04:46
【问题描述】:
我正在编写一个 python 程序,其中必须将大型命题公式转换为 z3 实例。例如公式f,我的程序可以通过
a = my_atomic_proposition("a") # Bool
b = my_atomic_proposition("b", operator.ge, 42) # Real: c >= 42
c = my_atomic_proposition("c") # Bool
f = my_and(a, my_or(b, my_not(c)))
应该转换成z3实例g即
a = z3.Bool("a")
b = z3.Real("b")
c = z3.Bool("c")
g = z3.And(a, z3.Or(b >= 42, z3.Not(c)))
请记住,我所说的公式主要包含 100 多个术语。
关注帖子
Z3 with string expressions 我首先尝试构建自己的解析器(Leornardo de Moura 在帖子中建议的选项 1),它只是递归地遍历我的所有公式的操作数并在途中构建 z3 实例。这个选项很慢,所以我尝试递归地构建更快的字符串,然后在它们上调用eval(上面帖子中描述的选项3)。这个解决方案要快得多,但是,当我的公式变得太大时(MemoryError 被抛出)就不起作用。
所以现在我要尝试第三个选项:使用z3.parse_smt2_string 从字符串创建z3 实例(不是我上面使用eval 的那个,它必须有不同的语法)。我会以某种方式进行,就像在Z3_parse_smtlib_string usage issues 中所做的那样。但是,我想知道使用z3.parse_smt2_string 是否会像使用eval 一样遇到类似的内存问题?从那时起,我可能会寻找另一种方式,然后再投入太多精力。
此外,如果有人对我如何实现我的目标有其他(希望是更聪明的想法),我会很高兴发表任何评论。谢谢你的帮助!
编辑 - MemoryError 示例:
我将举例说明抛出MemoryError 的情况:假设我有以下公式作为字符串:
f = 'z3.Or(a___0,z3.And(True,z3.Or(a___1,z3.And(True,z3.Or(a___2,z3.And(True,z3.Or(a___3,z3.And(True,z3.Or(a___4,z3.And(True,z3.Or(a___5,z3.And(True,z3.Or(a___6,z3.And(True,z3.Or(a___7,z3.And(True,z3.Or(a___8,z3.And(True,z3.Or(a___9,z3.And(True,z3.Or(a___10,z3.And(True,z3.Or(a___11,z3.And(True,z3.Or(a___12,z3.And(True,z3.Or(a___13,z3.And(True,z3.Or(a___14,z3.And(True,z3.Or(a___15,z3.And(True,z3.Or(a___16,z3.And(True,z3.Or(a___17,z3.And(True,z3.Or(a___18,z3.And(True,z3.Or(a___19,z3.And(True,z3.Or(a___20,z3.And(True,z3.Or(a___21,z3.And(True,z3.Or(a___22,z3.And(True,z3.Or(a___23,z3.And(True,z3.Or(a___24,z3.And(True,z3.Or(a___25,z3.And(True,z3.Or(a___26,z3.And(True,z3.Or(a___27,z3.And(True,z3.Or(a___28,z3.And(True,z3.Or(a___29,z3.And(True,z3.Or(a___30,z3.And(True,z3.Or(a___31,z3.And(True,z3.Or(a___32,z3.And(True,z3.Or(a___33,z3.And(True,z3.Or(a___34,z3.And(True,z3.Or(a___35,z3.And(True,z3.Or(a___36,z3.And(True,z3.Or(a___37,z3.And(True,z3.Or(a___38,z3.And(True,z3.Or(a___39,z3.And(True,z3.Or(a___40,z3.And(True,z3.Or(a___41,z3.And(True,z3.Or(a___42,z3.And(True,z3.Or(a___43,z3.And(True,z3.Or(a___44,z3.And(True,z3.Or(a___45,z3.And(True,z3.Or(a___46,z3.And(True,z3.Or(a___47,z3.And(True,z3.Or(a___48,z3.And(True,z3.Or(a___49,z3.And(True,a___50))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))'
以及以下代表f变量的字符串列表:
variables = ['a___2', 'a___31', 'a___34', 'a___46', 'a___29', 'a___12', 'a___9', 'a___32', 'a___41', 'a___24', 'a___38', 'a___23', 'a___19', 'a___50', 'a___3', 'a___6', 'a___35', 'a___28', 'a___13', 'a___16', 'a___0', 'a___33', 'a___36', 'a___40', 'a___45', 'a___10', 'a___39', 'a___43', 'a___22', 'a___27', 'a___7', 'a___49', 'a___21', 'a___17', 'a___1', 'a___4', 'a___37', 'a___44', 'a___11', 'a___14', 'a___30', 'a___42', 'a___47', 'a___8', 'a___26', 'a___48', 'a___20', 'a___25', 'a___5', 'a___15', 'a___18']
然后我以如下方式使用eval:
# Declare z3 variables for all strings in my list 'variables'
for x in variables:
globals()[x] = z3.Bool(x)
# Create z3 object from string 'f'
z3f = eval(f)
并收到错误:
---------------------------------------------------------------------------
MemoryError Traceback (most recent call last)
<ipython-input-71-b7421db475e5> in <module>()
3 globals()[x] = z3.Bool(x)
4 # Create z3 object from string 'f'
----> 5 z3f = eval(f)
MemoryError:
对于类似但较短的f,上面的代码可以正常工作。例如:
f = 'z3.Or(a___0,z3.And(True,z3.Or(a___1,z3.And(True,z3.Or(a___2,z3.And(True,z3.Or(a___3,z3.And(True,z3.Or(a___4,z3.And(True,z3.Or(a___5,z3.And(True,z3.Or(a___6,z3.And(True,z3.Or(a___7,z3.And(True,z3.Or(a___8,z3.And(True,z3.Or(a___9,z3.And(True,z3.Or(a___10,z3.And(True,z3.Or(a___11,z3.And(True,z3.Or(a___12,z3.And(True,z3.Or(a___13,z3.And(True,z3.Or(a___14,z3.And(True,z3.Or(a___15,z3.And(True,z3.Or(a___16,z3.And(True,z3.Or(a___17,z3.And(True,z3.Or(a___18,z3.And(True,z3.Or(a___19,z3.And(True,z3.Or(a___20,z3.And(True,z3.Or(a___21,z3.And(True,z3.Or(a___22,z3.And(True,z3.Or(a___23,z3.And(True,z3.Or(a___24,z3.And(True,z3.Or(a___25,z3.And(True,z3.Or(a___26,z3.And(True,z3.Or(a___27,z3.And(True,z3.Or(a___28,z3.And(True,z3.Or(a___29,z3.And(True,z3.Or(a___30,z3.And(True,z3.Or(a___31,z3.And(True,z3.Or(a___32,z3.And(True,z3.Or(a___33,z3.And(True,z3.Or(a___34,z3.And(True,z3.Or(a___35,z3.And(True,z3.Or(a___36,z3.And(True,z3.Or(a___37,z3.And(True,z3.Or(a___38,z3.And(True,z3.Or(a___39,z3.And(True,a___40))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))))'
variables = ['a___2', 'a___31', 'a___34', 'a___29', 'a___12', 'a___9', 'a___32', 'a___24', 'a___38', 'a___23', 'a___19', 'a___3', 'a___6', 'a___35', 'a___28', 'a___13', 'a___16', 'a___0', 'a___33', 'a___36', 'a___40', 'a___10', 'a___39', 'a___22', 'a___27', 'a___7', 'a___21', 'a___17', 'a___1', 'a___4', 'a___37', 'a___11', 'a___14', 'a___30', 'a___8', 'a___26', 'a___20', 'a___25', 'a___5', 'a___15', 'a___18']
我收到:
z3f = Or(a___0,
And(True,
Or(a___1,
And(True,
Or(a___2,
And(True,
Or(a___3,
And(True,
Or(a___4,
And(True,
Or(a___5,
And(True,
Or(a___6,
And(True,
Or(a___7,
And(True,
Or(a___8,
And(True,
Or(a___9,
And(True,
Or(..., ...)))))))))))))))))))))
【问题讨论】:
-
如果没有看到您要解析的实际公式,很难回答这个问题。如果没有这个,我会说你可能会看到 parse_smt2_string 的类似问题。然而,因为你得到一个 MemoryError,实际的问题也可能出在 Python 方面(Z3 从不自己引发这种类型的异常)。
-
@ChristophWintersteiger 我添加了一个实际公式的示例,我正在尝试解析抛出
MemoryError的位置。这很可能是 Python 本身的问题……