связать функцию как команду shell pipe в python - PullRequest
1 голос
/ 04 июня 2011

В оболочке Unix / Linux мы можем:

seq 0 100 | head -10 | awk 'NF%2==0' | awk 'NF%2==1' | rev

Теперь я определил:

seqsrc = list(range(0,100))

def all(src): return src
def head(src, count, offset = 0): return src[:count]
def tail(src, count, offset = 0): return src[-count:]
def odd(src): return [x for x in src if x % 2 != 0]
def even(src): return [x for x in src if x % 2 == 0]
def reverse(src): return src[::1]
...
#def other_sequence_manpulation_method()

Вот мои вопросы:

1. Как я могу получить shell shell как грамматику в python?

seqdst = all(seqsrc).head(10).odd().even().reverse()

2. По какой-то причине я хочу перечислить все возможные комбинации этих простых функций, которые я определил, могу ли я сделать это с помощью itertools.product () для генерации комбинаций - EDIT: а также для решения класса Seq ниже?

possible_head_limit = [10,20,30]

all(seqsrc).head(10)                    # 10 is one item in possible_head_limit
all(seqsrc).head(10).odd()
all(seqsrc).head(10).odd().even()
all(seqsrc).head(10).odd().even().reverse()
all(seqsrc).head(10).even()
all(seqsrc).head(10).even().odd()
....
all(seqsrc).head(20)                    # 20 is one item in possible_head_limit

all(seqsrc).head(20).odd()
...

3: Предположим, что seqsrc = range(0,10), тогда head(20) может вернуть то же самое, что и head(10), или когда-нибудь это будет бессмысленно

all(seqsrc).head(20).odd().even().reverse()  
# = all(seqsrc).head(10).odd().even().reverse() 
# = all(seqsrc).head(11).odd().even().reverse()
# ...

Могу ли я добавить функцию управления в цепочку методов, тогда я могу контролировать среднее значение возврата?

ignore_insufficient(True).all(seqsrc).head(20).odd().even().reverse()
ignore_insufficient(False).all(seqsrc).head(20).odd().even().reverse()  # it will print some sort of error

# or even I can control each function I defined?
ignore_insufficient(True).all(seqsrc).\
ignore_insufficient(True).head(20).\
ignore_insufficient(False).tail(10)

Спасибо!

1 Ответ

4 голосов
/ 04 июня 2011

В вашем сообщении много вопросов, и я не уверен, что понимаю их все. Тем не менее, вот отправная точка.

Цепные методы обычно реализуются путем разработки классов с методами, которые возвращают новые экземпляры самого класса. Это позволяет вызывать дополнительные методы из возвращаемого значения предыдущих методов.

Следовательно, вы можете определить класс Seq следующим образом:

class Seq(object):
    def __init__(self, seq):
        self.seq = seq
    def __repr__(self):
        return repr(self.seq)
    def __str__(self):
        return str(self.seq)
    def all(self):
        return Seq(self.seq[:])
    def head(self, count):
        return Seq(self.seq[:count])
    def tail(self, count):
        return Seq(self.seq[-count:])
    def odd(self):
        return Seq(self.seq[1::2])
    def even(self):
        return Seq(self.seq[::2])
    def reverse(self):
        return Seq(self.seq[::-1])

И используйте это как:

>>> s = Seq(range(0, 100))
>>> print s.head(10).odd().even().reverse()
[9, 5, 1]

Обратите внимание, что это можно улучшить многими способами. Например, в мире Javascript цепочечные методы jQuery фактически помещают свои результаты в стек, что позволяет возвращать в историю вызовов и восстанавливать предыдущий контекст. Подробнее см. end () .

...