可以将文章内容翻译成中文,广告屏蔽插件可能会导致该功能失效(如失效,请关闭广告屏蔽插件后再试):
问题:
This question already has an answer here:
-
how to interleaving lists [duplicate]
2 answers
I want to be able to interleave two lists that could potentially be unequal in length. What I have is:
def interleave(xs,ys):
a=xs
b=ys
c=a+b
c[::2]=a
c[1::2]=b
return c
This works great with lists that either equal in length or just +/-1. But if let's say xs=[1,2,3] and ys= ["hi,"bye","no","yes","why"] this message appears:
c[::2]=a
ValueError: attempt to assign sequence of size 3 to extended slice of size 4
How do I fix this with using indexing? or do I have to use for loops?
EDIT: what I want is to have the extra values just appear at the end.
回答1:
You can use itertools.izip_longest
here:
>>> from itertools import izip_longest
>>> xs = [1,2,3]
>>> ys = ["hi","bye","no","yes","why"]
>>> s = object()
>>> [y for x in izip_longest(xs, ys, fillvalue=s) for y in x if y is not s]
[1, 'hi', 2, 'bye', 3, 'no', 'yes', 'why']
Using roundrobin
recipe from itertools, no sentinel value required here:
from itertools import *
def roundrobin(*iterables):
"roundrobin('ABC', 'D', 'EF') --> A D E B F C"
# Recipe credited to George Sakkis
pending = len(iterables)
nexts = cycle(iter(it).next for it in iterables)
while pending:
try:
for next in nexts:
yield next()
except StopIteration:
pending -= 1
nexts = cycle(islice(nexts, pending))
Demo:
>>> list(roundrobin(xs, ys))
[1, 'hi', 2, 'bye', 3, 'no', 'yes', 'why']
>>> list(roundrobin(ys, xs))
['hi', 1, 'bye', 2, 'no', 3, 'yes', 'why']
回答2:
You can use heapq.merge
:
xs = [1, 2, 3]
ys = ['hi', 'bye', 'no', 'yes', 'why']
import heapq
interleaved = [v for i, v in heapq.merge(*[enumerate(el) for el in (xs, ys)])]
# [1, 'hi', 2, 'bye', 3, 'no', 'yes', 'why']
This avoids the need for a sentinel value and flattening.
Use the roundrobin recipe instead to achieve this more effectively without having items be comparable.
回答3:
Okay, here's my entry:
>>> from itertools import chain
>>> xs = [1,2,3]
>>> ys = ["hi","bye","no","yes","why"]
>>> xi, yi = iter(xs), iter(ys)
>>> list(chain.from_iterable(zip(xi, yi))) + list(xi) + list(yi)
[1, 'hi', 2, 'bye', 3, 'no', 'yes', 'why']
Alternatively,
>>> [i for row in zip(xi, yi) for i in row] + list(xi) + list(yi)
would have worked too (that's just the listcomp idiom for flattening, as used by @hcwhsa). My first thought was
>>> list(zip(*sorted(list(enumerate(xs)) + list(enumerate(ys)))))[1]
(1, 'hi', 2, 'bye', 3, 'no', 'yes', 'why')
but that's just a much less efficient version of @Jon Clements (I used an inefficient sort, he used an efficient heap queue.)
[I've been experimenting trying to get something using cycle
working, but it doesn't seem to be as easy as I'd hoped: and it turns out that I was simply working toward reimplementing the roundrobin recipe that @hcwsha posted, so there's no point in finishing that. :^) ]
回答4:
Kept it simple:
def interleave(xs,ys):
stop = min(len(xs), len(ys))
suffix = max(xs, ys, key = len)[stop:]
out = list()
for pair in zip(xs, ys):
out.extend(pair)
out.extend(suffix)
return out
Caveats:
Python 2.7
Assumes lists are passed as the arguments.