Чи є в Python хороший спосіб чергування двох списків однакової довжини?
Скажіть, мені дано [1,2,3]і [10,20,30]. Я хотів би перетворити їх на [1,10,2,20,3,30].
Відповіді:
Опублікувавши питання, я зрозумів, що можу просто зробити наступне:
[val for pair in zip(l1, l2) for val in pair]
де l1і l2- два списки.
Якщо існує N списків для чергування, тоді
lists = [l1, l2, ...]
[val for tup in zip(*lists) for val in tup]
izip_longestдля python2 та zip_longestдля python3 ` [val for pair in itertools.zip_longest(l1, l2) for val in pair]результати з['a', 'b', 'a', 'b', 'a', 'b', None, 'b', None, 'b', None, 'b']
Для Python> = 2.3 існує розширений синтаксис зрізів :
>>> a = [0, 2, 4, 6, 8]
>>> b = [1, 3, 5, 7, 9]
>>> c = a + b
>>> c[::2] = a
>>> c[1::2] = b
>>> c
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
Рядок c = a + bвикористовується як простий спосіб створити новий список точно потрібної довжини (на цьому етапі його зміст не важливий). Наступні два рядки роблять фактичну роботу перемежения aі b: перший привласнює елементи aдля всіх парних індексів c; другий присвоює елементи bвсім непарним індексам c.
Дано
a = [1, 2, 3]
b = [10, 20, 30]
c = [100, 200, 300, 999]
Код
Припускаючи списки однакової довжини, ви можете отримати перемежований список з itertools.chainі zip:
import itertools
list(itertools.chain(*zip(a, b)))
# [1, 10, 2, 20, 3, 30]
Альтернативи
Більш загально з нерівними списками використовуйте zip_longest(рекомендується):
[x for x in itertools.chain(*itertools.zip_longest(a, c)) if x is not None]
# [1, 100, 2, 200, 3, 300, 999]
Багато списків можна безпечно чергувати:
[x for x in itertools.chain(*itertools.zip_longest(a, b, c)) if x is not None]
# [1, 10, 100, 2, 20, 200, 3, 30, 300, 999]
Бібліотека, яка постачається з roundrobinрецептом itertools, interleaveта interleave_longest.
import more_itertools
list(more_itertools.roundrobin(a, b))
# [1, 10, 2, 20, 3, 30]
list(more_itertools.interleave(a, b))
# [1, 10, 2, 20, 3, 30]
list(more_itertools.interleave_longest(a, c))
# [1, 100, 2, 200, 3, 300, 999]
yield from
Нарешті, для чогось цікавого в Python 3 (хоча і не рекомендується):
list(filter(None, ((yield from x) for x in zip(a, b))))
# [1, 10, 2, 20, 3, 30]
list([(yield from x) for x in zip(a, b)])
# [1, 10, 2, 20, 3, 30]
+ Встановити за допомогоюpip install more_itertools
Мені потрібен був спосіб зробити це зі списками різного розміру, яких прийнята відповідь не стосується.
Моє рішення використовує генератор, і його використання виглядає трохи приємніше через це:
def interleave(l1, l2):
iter1 = iter(l1)
iter2 = iter(l2)
while True:
try:
if iter1 is not None:
yield next(iter1)
except StopIteration:
iter1 = None
try:
if iter2 is not None:
yield next(iter2)
except StopIteration:
iter2 = None
if iter1 is None and iter2 is None:
raise StopIteration()
І його використання:
>>> a = [1, 2, 3, 4, 5]
>>> b = ['a', 'b', 'c', 'd', 'e', 'f', 'g']
>>> list(interleave(a, b))
[1, 'a', 2, 'b', 3, 'c', 4, 'd', 5, 'e', 'f', 'g']
>>> list(interleave(b, a))
['a', 1, 'b', 2, 'c', 3, 'd', 4, 'e', 5, 'f', 'g']
roundrobinРецепт від itertoolsмодуля є більш загальним розширенням цього.
Альтернатива:
>>> l1=[1,2,3]
>>> l2=[10,20,30]
>>> [y for x in map(None,l1,l2) for y in x if y is not None]
[1, 10, 2, 20, 3, 30]
Це працює, оскільки карта працює над списками паралельно. Це працює однаково під 2.2. Сам по собі, Noneяк викликані функції, mapстворює список кортежів:
>>> map(None,l1,l2,'abcd')
[(1, 10, 'a'), (2, 20, 'b'), (3, 30, 'c'), (None, None, 'd')]
Потім просто розрівняйте список кортежів.
Звичайно, перевага map буде працювати для будь-якої кількості списків і працюватиме, навіть якщо вони різної довжини:
>>> l1=[1,2,3]
>>> l2=[10,20,30]
>>> l3=[101,102,103,104]
>>> [y for x in map(None,l1,l2,l3) for y in x if y in not None]
[1, 10, 101, 2, 20, 102, 3, 30, 103, 104]
if yтакож фільтрується 0, if y is not Noneменш крихкий.
Мені найбільше подобається рішення Aix. ось ще один спосіб, на мою думку, повинен працювати в 2.2:
>>> x=range(3)
>>> x
[0, 1, 2]
>>> y=range(7,10)
>>> y
[7, 8, 9]
>>> sum(zip(x,y),[])
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: can only concatenate list (not "tuple") to list
>>> sum(map(list,zip(x,y)),[])
[0, 7, 1, 8, 2, 9]
і ще один спосіб:
>>> a=[x,y]
>>> [a[i][j] for j in range(3) for i in (0,1)]
[0, 7, 1, 8, 2, 9]
і:
>>> sum((list(i) for i in zip(x,y)),[])
[0, 7, 1, 8, 2, 9]
[el for el in itertools.chain(*itertools.izip_longest([1,2,3], [4,5])) if el is not None]
Поки у вас немає того, Noneщо ви хочете зберегти
Щоб відповісти на назву запитання "Перемішати кілька списків однакової довжини в Python", ми можемо узагальнити відповідь із 2 списків @ekhumoro. Це явно вимагає, щоб списки були однакової довжини, на відміну від (елегантного) рішення @NPE
import itertools
def interleave(lists):
"""Interleave a list of lists.
:param lists: List of lists; each inner length must be the same length.
:returns: interleaved single list
:rtype: list
"""
if len(set(len(_) for _ in lists)) > 1:
raise ValueError("Lists are not all the same length!")
joint = list(itertools.chain(*lists))
for l_idx, li in enumerate(lists):
joint[l_idx::len(lists)] = li
return joint
Приклади:
>>> interleave([[0,2,4], [1, 3, 5]])
[0, 1, 2, 3, 4, 5]
>>> interleave([[0,2,4], [1, 3, 5], [10, 11, 12]])
[0, 1, 10, 2, 3, 11, 4, 5, 12]
>>> interleave([[0,2,4], [1, 3, 5], [10, 11, 12], [13, 14, 15]])
[0, 1, 10, 13, 2, 3, 11, 14, 4, 5, 12, 15]
>>> interleave([[0,2,4], [1, 3, 5], [10, 11, 12], [13, 14]])
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 10, in interleave
ValueError: Lists are not all the same length!
>>> interleave([[0,2,4]])
[0, 2, 4]
Занадто пізно на вечірку, і є багато хороших відповідей, але я також хотів би запропонувати просте рішення, використовуючи extend()метод:
list1 = [1, 2, 3]
list2 = [10, 20, 30]
new_list = []
for i in range(len(list1)):
new_list.extend([list1[i], list2[i]])
print(new_list)
Вихід:
[1, 10, 2, 20, 3, 30]
it = iter(l1); list((yield next(it)) or i for i in l2)