或者,“在生产中永远不要做的事情”
前几天,我正在看一些Python代码,这些代码是由一系列深层嵌套的函数调用组成的,一个想法立即被推高了大脑:
如果Python具有类似于Elixir的管道运算符的运算符,这会很酷吗?
管道操作员?
对于不熟悉Elixir的用户,可以使用管道运算符|>
来转换如下代码:
变成看起来像这样的代码:
本质上,假设右侧的表达式是函数调用,则管道运算符在左侧使用表达式并将其移动到右侧的表达式的第一个参数位置。
用Python做
尽管不更改Python解释器就不可能在语言中添加实际的管道运算符,但是没有什么可以阻止我们重新使用现有的运算符! 这正是我着手要做的。
从shell借来的东西,我认为“按位或”(又名“管道”)运算符将非常适合这种功能:
重新构想管道运算符的最简单方法是使用装饰器重写函数。 Python的内置ast
模块使此操作特别容易。
我要做的就是
- 获取函数的源代码(使用
inspect.getsource
), - 通过将其传递给
ast.parse
,将其转变为抽象语法树 - 根据我在上一节末尾概述的规则,在树上走动并变换管道操作符的所有出现,
- 最后,编译并返回重写的函数对象。
这听起来比实际要复杂得多。
ast
模块提供了一个称为NodeTransformer
的类,该类实现了访问者模式:它的visit
方法通过AST进行深度优先搜索,并为树中的每个节点自行调用visit_NODETYPE
形式的任何已声明方法。 顾名思义,您可以使用节点转换器来访问和操纵AST的节点:
每当转换器遇到二进制运算符时,它都会递归,以便首先进行需要在其左侧节点和右侧节点上进行的所有转换,然后检查当前运算符是否为“按位或”。
如果当前节点的运算符是“按位或”,并且右侧是函数调用节点,则它将左侧插入函数调用的第一个参数位置,然后返回右侧的节点方面,将二元运算符节点替换为树中的调用节点。
转换器在看到函数定义时也会启动,以便可以删除enable_threadop
装饰器。 看一下装饰器本身,这将很有意义:
装饰器接受一个函数作为参数,获取其源代码并删除所有缩进(这很重要!否则,装饰类方法将导致SyntaxError
),解析该代码并转换AST,最后编译并执行该函数定义,返回结果函数对象。
如果转换器没有从最后一棵树中删除装饰器,那么我们将有一个无限循环,因为在执行函数定义时,一遍enable_threadop
调用enable_threadop
(第13行)。
一切就绪后,可以使用enable_threadop
装饰器有选择地更改管道运算符的行为:
局限性
如您所料,此方法有两个限制。
首先, inspect.getsource
从文件系统中inspect.getsource
函数的源代码,这意味着装饰器将无法在Python解释器中工作。
其次,转换器要求管道操作员的右侧是函数调用。
是时候在生产中使用它了!
哇,慢点! 这只是一个小巧的实验,绝对不是您应该对同事施加的任何东西!
就是说,如果您想使用它,可以在GitHub上找到完整的代码(带有注释的50行!)。
谢谢阅读! 如果您喜欢这篇文章,请拍一下! 您也可以在我的网站 https : //defn.io,GitHub 和 Twitter 上 找到我 。
Bogdan Popa(@bogdanp)的最新推文。 程序员,https: //t.co/FFd6cPhKk5 和… twitter.com的创建者
From: https://hackernoon.com/adding-a-pipe-operator-to-python-19a3aa295642