Как закрепить два списка по-разному?
Я хочу закрепить два списка с разной длиной
например
A = [1,2,3,4,5,6,7,8,9]
B = ["A","B","C"]
и я ожидаю этого
[(1, 'A'), (2, 'B'), (3, 'C'), (4, 'A'), (5, 'B'), (6, 'C'), (7, 'A'), (8, 'B'), (9, 'C')]
Но встроенный zip
не будет повторять пару со списком большего размера.
Существует ли какой-либо встроенный способ, который может достичь этого?
спасибо
вот мой код
idx = 0
zip_list = []
for value in larger:
zip_list.append((value,smaller[idx]))
idx += 1
if idx == len(smaller):
idx = 0
Ответы
Ответ 1
Вы можете использовать itertools.cycle
:
Сделать итератор, возвращающий элементы из итерабельного и сохраняя копию каждого. Когда итерируемый будет исчерпан, верните элементы из сохраненной копии. Повторяет бесконечно.
Пример:
A = [1,2,3,4,5,6,7,8,9]
B = ["A","B","C"]
from itertools import cycle
zip_list = zip(A, cycle(B)) if len(A) > len(B) else zip(cycle(A), B)
Ответ 2
Попробуйте это.
A = [1,2,3,4,5,6,7,8,9]
B = ["A","B","C"]
Z = []
for i, a in enumerate(A):
Z.append((a, B[i % len(B)]))
Просто убедитесь, что более крупный список находится в A
.
Ответ 3
Вероятно, есть лучший способ, но вы можете сделать функцию, которая повторяет ваш список до любой длины.
def repeatlist(l,i):
'''give a list and a total length'''
while len(l) < i:
l += l
while len(l) > i:
l.pop()
Тогда do
repeatlist(B,len(A))
zip_list = zip(A,B)
Ответ 4
симметричный, без условных условий один вкладыш
[*zip(A*(len(B)//len(A) + 1), B*(len(A)//len(B) + 1))]
который строго отвечает: "Как закрепить два списка по-разному?"
требуется патч для списков одинакового размера:
[*(zip(A, B) if len(A) == len(B)
else zip(A*(len(B)//len(A) + 1),
B*(len(A)//len(B) + 1)))]
Ответ 5
И в настоящее время с перечислениями в списке
[(i, B[i % 3 - 1]) for i in A]
Или, если элементы A
не являются последовательными и не беспокоятся о длинах списков
[(j, B[i % len(B)]) for i, j in enumerate(A)] if len(A) >= len(B) else \
[(A[i % len(A)], j) for i, j in enumerate(B)]
Ответ 6
Для версии, которая работает с любым конечным числом потенциально бесконечных итераций в любом порядке:
from itertools import cycle, tee, zip_longest
def cyclical_zip(*iterables):
iterables_1, iterables_2 = zip(*map(tee, iterables)) # Allow proper iteration of iterators
for _, x in zip(
zip_longest(*iterables_1), # Limit by the length of the longest iterable
zip(*map(cycle, iterables_2))): # the cycling
yield x
assert list(cyclical_zip([1, 2, 3], 'abcd', 'xy')) == [(1, 'a', 'x'), (2, 'b', 'y'), (3, 'c', 'x'), (1, 'd', 'y')] # An example and test case
Ответ 7
Решение для произвольного числа итераций, и вы не знаете, какой из них самый длинный (также допускается использование по умолчанию для любых пустых итераций):
from itertools import cycle, zip_longest
def zip_cycle(*iterables, empty_default=None):
cycles = [cycle(i) for i in iterables]
for _ in zip_longest(*iterables):
yield tuple(next(i, empty_default) for i in cycles)
for i in zip_cycle(range(2), range(5), ['a', 'b', 'c'], []):
print(i)
Выходы:
(0, 0, 'a', None)
(1, 1, 'b', None)
(0, 2, 'c', None)
(1, 3, 'a', None)
(0, 4, 'b', None)