Pipe——Python 的中缀语法库

翻译 2011年03月29日 20:35:00

赖勇浩(http://laiyonghao.com)

注:本文基本上是翻译这篇文章(http://dev-tricks.net/pipe-infix-syntax-for-python)。

通过 Pipe 模块,就能够使用 Python 用上中缀语法。

首先来看一下传统前缀语法的代码:

sum(select(where(take_while(fib(), lambda x: x < 1000000) lambda x: x % 2), lambda x: x * x))

很难读?再来看看中缀语法代码:

fib() | take_while(lambda x: x < 1000000) \
      | where(lambda x: x % 2) \
      | select(lambda x: x * x) \
      | sum()

好读多了吧?

虽然 Pipe 基类的代码很少,但很强大,能够让你很容易写出 pipeable 函数哦。而且这个模块本身就带了超过 30 个已经写好的函数,比如 ‘where’, ‘group_by’, ‘sort’, ‘take_while’ …

如果想一下 Pipe,需要先安装,在命令行执行:

pip install -U pipe

然后等着安装完成就行了。现在可以打开一个交互式 Shell,来试一下:

Python 2.6.6 (r266:84292, Dec 26 2010, 22:31:48) 
[GCC 4.4.5] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> from pipe import *
>>> [1, 2, 3, 4, 5] | add
15
>>> [5, 4, 3, 2, 1] | sort
[1, 2, 3, 4, 5]

很简单吧?如果有什么问题,可以随时 help(pipe) 一下,就可以看到完备的帮助了。

接下来我们展示一下组合两个或更多的 pipeable 函数:

>>> [1, 2, 3, 4, 5] | where(lambda x: x % 2) | concat
'1, 3, 5'
>>> [1, 2, 3, 4, 5] | where(lambda x: x % 2) | tail(2) | concat
'3, 5'
>>> [1, 2, 3, 4, 5] | where(lambda x: x % 2) | tail(2) | select(lambda x: x * x) | concat
'9, 25'
>>> [1, 2, 3, 4, 5] | where(lambda x: x % 2) | tail(2) | select(lambda x: x * x) | add
34

因为 Pipe 是惰性求值的,所以我们完成可以弄一个无穷生成器而不用担心内存用完,比如:

>>> def fib():
...    a, b = 0, 1
...    while True:
...        yield a
...        a, b = b, a + b

现在让我们用 fib() 函数来完成一个 http://projecteuler.net 的第 2 题:

Find the sum of all the even-valued terms in Fibonacci which do not exceed four million.

>>> euler2 = fib() | where(lambda x: x % 2 == 0) | take_while(lambda x: x < 4000000) | add
>>> assert euler2 == 4613732

怎么样?可读性强吧?漂亮不?

最后,我们来学习一下如何利用 @Pipe decorator 创建一个新的 pipeable 函数:

假定要创建一个函数 yield 它的输入的前 x 个元素

假定要创建一个函数能够用以 (1, 2, 3, 4, 5) | take(2) 语句来获取前两个元素

那么最初的实现可能是这样的:

def take(iterable, qte):
    for item in iterable:
        if qte > 0:
            qte -= 1
            yield item
        else:
            return

现在,你只要把 @Pipe 盖在这个函数上头,这货就是 pipeable 函数了!

====================

鸣谢:

感谢 @yinhm 在 Twitter 上分享《Pipe: Infix syntax for Python》一文,让我知道还有这等神器。

感谢 @kyhpudding 在 Twitter 上分享他的 solo 模块,一个比 pipe 更奇幻的模块,希望我能理解清楚,能够跟大家介绍之。

[Python]中缀表达式转前缀表达式

#判断运算符的优先级 def opOrder(op1,op2): order_dic = {'*':4,'$':5,'/':4,'+':3,'-':3} if op1 == '(' o...
  • sinat_16968575
  • sinat_16968575
  • 2015-03-20 19:05:06
  • 1029

python数据结构与算法 9 中缀后前缀、后缀的转换思路

中缀到前缀和后缀的转换 So far, we have used adhoc methods to convert between infix expressions and the equival...
  • ppabcde
  • ppabcde
  • 2014-03-16 09:21:00
  • 2119

python数据结构与算法 8栈的应用之中缀前缀后缀

python算法与数据结构,栈的应用,前缀后缀中缀的概念,下一节讲到互相转换。...
  • ppabcde
  • ppabcde
  • 2014-03-16 08:06:10
  • 1469

Python算法库安装简介

安装python算法库 Python算法库的安装非常简单: 执行C:\Python64\Scripts\pip install库名(小写字母不加后缀) Python 算法库的安装顺序:Numpy S...
  • u013285384
  • u013285384
  • 2016-08-17 19:26:41
  • 1592

python安装算法库

C:\Python27\Scripts\pip install numpy列出所有已安装库: C:\Python27\Scripts\pip list
  • daijiguo
  • daijiguo
  • 2016-05-31 09:40:28
  • 1056

C语言后缀式转中缀式的计算代码

  • 2012年04月05日 00:32
  • 3KB
  • 下载

python实现中缀表达式计算,实现指数和对数

import os,sys,decimal,math def stack_sum_pop(sign,sum): t2,t3=0,0 t1=sign.pop() sum_a=...
  • qq_40164942
  • qq_40164942
  • 2018-01-05 09:32:16
  • 44

python数据结构与算法 10 栈的应用之中缀转后缀表达式算法的实现

General Infix-to-Postfix Conversion 中缀转后缀的算法 We need todevelop an algorithm to convert any infix exp...
  • ppabcde
  • ppabcde
  • 2014-03-16 15:24:58
  • 1962

利用Python实现中缀表达式转后缀表达式

这里我们写一个Python的程序来实现将一个算术表达式(中缀表达式)如何转换成一个后缀表达式。由于初学Python,所以这里实现的算术符可能就只有基本的算式运算符,但是思想还是共同的,值得借鉴参考。 ...
  • minxihou
  • minxihou
  • 2016-06-19 23:27:14
  • 2762

matplotlib-1.5.1.tar.gz

  • 2017年01月08日 17:20
  • 51.53MB
  • 下载
收藏助手
不良信息举报
您举报文章:Pipe——Python 的中缀语法库
举报原因:
原因补充:

(最多只允许输入30个字)