正确地从字符串**创建一个lambda函数**

Hug*_*ell 13 python string lambda

给出一个字符串如

"2*(i+j) <= 100"
Run Code Online (Sandbox Code Playgroud)

我想生成相应的lambda函数,

fn = lambda i,j: 2*(i+j) <= 100
Run Code Online (Sandbox Code Playgroud)
  • 我可以这样做eval,但我正在寻找一种不那么邪恶的方法.

  • 我已经找到

    import ast
    f = ast.Lambda('i,j', '2*(i+j) <= 100')
    
    Run Code Online (Sandbox Code Playgroud)

    但我还没弄清楚如何执行结果!

  • 理想情况下,我想自动拉出参数列表('i','j') - 现在,我只是使用re.findall('\ w +'),但我希望能够正确使用现有的功能,cos而不是将它们视为"关键字".


我在看是否有用于处理复杂数学集的Python库(使用数学集合构建器表示法构建)?并试图弄清楚如何最好地将set-builder符号解析为lambdas以提供给约束求解器.

我基本上希望ast.literal_eval也可以识别变量.

理想情况下,鉴于i >= 20我想回来((lambda x: x >= 20), ['i']),然后我可以直接提供给constraint.

Ned*_*der 14

你正在寻找替代方案eval,但为什么呢?你接受任意代码并执行它,所以为什么不使用eval?要避免的唯一原因eval是因为它很危险,但你最终创造的lambda也同样危险.

另外,请记住,在CPython中执行此操作确实无法安全

  • **无法完成**是一个很少适用于Python的短语.*ast.literal_eval*函数是一个很好的例子,说明如何评估任意代码,同时限制你可以接受的内容.此外,如果OP使用*trusted*input,那么*eval*或*exec*是完全合理的(例如,Guido在*timeit*模块中使用它们). (4认同)

Ray*_*ger 4

如果您的输入来自可信来源,则eval()是最简单、最清晰且最可靠的方法。

如果您的输入不受信任,则需要对其进行清理

一种合理的方法是使用正则表达式。确保字符串中没有函数调用、属性查找或双下划线。

或者,更复杂的方法是遍历 AST 解析树以确定是否存在任何令人反感的调用。

第三种方法是遍历 AST 解析树并直接执行它。这使您可以完全控制接到的电话。ast.literal_eval 函数采用这种方法。也许您从其源代码开始,并对您想要支持的任何操作进行一些构建:

def literal_eval(node_or_string):
    """
    Safely evaluate an expression node or a string containing a Python
    expression.  The string or node provided may only consist of the following
    Python literal structures: strings, numbers, tuples, lists, dicts, booleans,
    and None.
    """
    _safe_names = {'None': None, 'True': True, 'False': False}
    if isinstance(node_or_string, basestring):
        node_or_string = parse(node_or_string, mode='eval')
    if isinstance(node_or_string, Expression):
        node_or_string = node_or_string.body
    def _convert(node):
        if isinstance(node, Str):
            return node.s
        elif isinstance(node, Num):
            return node.n
        elif isinstance(node, Tuple):
            return tuple(map(_convert, node.elts))
        elif isinstance(node, List):
            return list(map(_convert, node.elts))
        elif isinstance(node, Dict):
            return dict((_convert(k), _convert(v)) for k, v
                        in zip(node.keys, node.values))
        elif isinstance(node, Name):
            if node.id in _safe_names:
                return _safe_names[node.id]
        elif isinstance(node, BinOp) and \
             isinstance(node.op, (Add, Sub)) and \
             isinstance(node.right, Num) and \
             isinstance(node.right.n, complex) and \
             isinstance(node.left, Num) and \
             isinstance(node.left.n, (int, long, float)):
            left = node.left.n
            right = node.right.n
            if isinstance(node.op, Add):
                return left + right
            else:
                return left - right
        raise ValueError('malformed string')
    return _convert(node_or_string)
Run Code Online (Sandbox Code Playgroud)