Python相当于Ruby的#each_cons?

max*_*ins 14 ruby python enumerable equivalent

有没有Pythonic等同于Ruby的#each_cons

在Ruby中你可以这样做:

array = [1,2,3,4]
array.each_cons(2).to_a
=> [[1,2],[2,3],[3,4]]
Run Code Online (Sandbox Code Playgroud)

Gus*_*son 17

我不认为有一个,我查看了内置模块itertools,这是我期望的.你可以简单地创建一个:

def each_cons(x, size):
    return [x[i:i+size] for i in range(len(x)-size+1)]
Run Code Online (Sandbox Code Playgroud)

  • 通用答案+1。我的适用于“2”,但在将其修改为任意“cons”后,它看起来像你的。 (2认同)

Eli*_*sky 11

对于这样的事情,itertools您应该关注的模块是:

from itertools import tee, izip

def pairwise(iterable):
    "s -> (s0,s1), (s1,s2), (s2, s3), ..."
    a, b = tee(iterable)
    next(b, None)
    return izip(a, b)
Run Code Online (Sandbox Code Playgroud)

然后:

>>> list(pairwise([1, 2, 3, 4]))
[(1, 2), (2, 3), (3, 4)]
Run Code Online (Sandbox Code Playgroud)

有关更一般的解决方案,请考虑以下事项:

def split_subsequences(iterable, length=2, overlap=0):
    it = iter(iterable)
    results = list(itertools.islice(it, length))
    while len(results) == length:
        yield results
        results = results[length - overlap:]
        results.extend(itertools.islice(it, length - overlap))
    if results:
        yield results
Run Code Online (Sandbox Code Playgroud)

这允许任意长度的子序列和任意重叠.用法:

>> list(split_subsequences([1, 2, 3, 4], length=2))
[[1, 2], [3, 4]]
>> list(split_subsequences([1, 2, 3, 4], length=2, overlap=1))
[[1, 2], [2, 3], [3, 4], [4]]
Run Code Online (Sandbox Code Playgroud)


sni*_*nip 6

我的列表解决方案(Python2):

import itertools
def each_cons(xs, n):
    return itertools.izip(*(xs[i:] for i in xrange(n)))
Run Code Online (Sandbox Code Playgroud)

编辑:itertools.izip不再使用Python 3 ,所以你使用plain zip:

def each_cons(xs, n):
    return zip(*(xs[i:] for i in range(n)))
Run Code Online (Sandbox Code Playgroud)


Ble*_*der 5

快速一句:

a = [1, 2, 3, 4]

out = [a[i:i + 2] for i in range(len(a) - 1)]
Run Code Online (Sandbox Code Playgroud)


dco*_*ish 5

Python 肯定可以做到这一点。如果你不想这么急切,可以使用 itertool 的 islice 和 izip。另外,重要的是要记住,普通切片将创建一个副本,因此如果内存使用很重要,您还应该考虑 itertool 等效项。

each_cons = lambda l: zip(l[:-1], l[1:])


Eli*_*les 5

更新:toolz.itertoolz.sliding_window()不要介意我下面的答案,只需使用- 它会做正确的事情。


对于each_cons在序列/生成器长度不足时保留 Ruby 行为的真正惰性实现:

import itertools
def each_cons(sequence, n):
    return itertools.izip(*(itertools.islice(g, i, None)
                          for i, g in
                          enumerate(itertools.tee(sequence, n))))
Run Code Online (Sandbox Code Playgroud)

例子:

>>> print(list(each_cons(xrange(5), 2)))
[(0, 1), (1, 2), (2, 3), (3, 4)]
>>> print(list(each_cons(xrange(5), 5)))
[(0, 1, 2, 3, 4)]
>>> print(list(each_cons(xrange(5), 6)))
[]
>>> print(list(each_cons((a for a in xrange(5)), 2)))
[(0, 1), (1, 2), (2, 3), (3, 4)]
Run Code Online (Sandbox Code Playgroud)

请注意,用于 izip 参数的元组解包应用于大小nitertools.tee(xs, n)(即“窗口大小”)的元组,而不是我们想要迭代的序列。