Як об'єднати списки в список кортежів?


297

Який пітонічний підхід для досягнення наступного?

# Original lists:

list_a = [1, 2, 3, 4]
list_b = [5, 6, 7, 8]

# List of tuples from 'list_a' and 'list_b':

list_c = [(1,5), (2,6), (3,7), (4,8)]

Кожен член list_c- кортеж, перший член якого є, list_aа другий - з list_b.

Відповіді:


442

У Python 2:

>>> list_a = [1, 2, 3, 4]
>>> list_b = [5, 6, 7, 8]
>>> zip(list_a, list_b)
[(1, 5), (2, 6), (3, 7), (4, 8)]

У Python 3:

>>> list_a = [1, 2, 3, 4]
>>> list_b = [5, 6, 7, 8]
>>> list(zip(list_a, list_b))
[(1, 5), (2, 6), (3, 7), (4, 8)]

77
ви повинні знати, що функція zip зупиняється в кінці найкоротшого списку, що може бути не завжди таким, яким ви хочете. itertoolsмодуль визначає zip_longest()метод , який зупиняється в кінці довгого списку, заповнюючи відсутні значення , з чим - то ви надаєте в якості параметра.
Адріан Пліссон

5
@Adrien: вітайте за ваш відповідний коментар. Для Python 2.x, s/zip_longest()/izip_longest(). Перейменований в Python 3.x to zip_longest().
mechanical_meat

Чи можу я створити [(1,5), (1,6), (1,7), (1,8), (2,5), (2,6) тощо] за допомогою команди zip?
Мона Джалал

3
@MonaJalal: ні, це не спарювання, це створення продукту списків. itertools.product()робить це.
Martijn Pieters

2
зауважте, щонайменше, в python3.6 zip не повертає список. Тож вам потрібен список (zip (list_a, list_b)) замість цього
Supamee

141

У python 3.0 zip повертає об'єкт zip. Ви можете отримати список із нього, зателефонувавши list(zip(a, b)).


3
Це може бути тривіально, але застережте, що використання цього напряму в циклі for дає вам генератор, який буде вичерпаний після використання одного разу. Збережіть змінну, якщо хочете використовувати її частіше
Hakaishin

13

Ви можете використовувати карту лямбда

a = [2,3,4]
b = [5,6,7]
c = map(lambda x,y:(x,y),a,b)

Це також спрацює, якщо довжина оригінальних списків не збігається


1
Навіщо використовувати лямбда? map(None, a,b)
Padraic Cunningham

У мене є лише доступ до python 3.5.
Темний лицар

3
Якщо ви використовували python3, тоді c не буде списком, це був би об'єкт карти, також використання лямбда буде набагато менш ефективним, ніж просто перетягування, якщо у вас є різні списки довжини і хочете обробити це, то ви б використовували izip_lo most / zip_lo most
Cunningham


6

Я не впевнений, це пітонічний шлях чи ні, але це здається простим, якщо обидва списки мають однакову кількість елементів:

list_a = [1, 2, 3, 4]

list_b = [5, 6, 7, 8]

list_c=[(list_a[i],list_b[i]) for i in range(0,len(list_a))]

5

Я знаю, що це старе питання, і на нього вже відповіли, але чомусь я все-таки хочу опублікувати це альтернативне рішення. Я знаю, що легко дізнатися, яка вбудована функція виконує потрібну вам "магію", але не шкода знати, що ви можете це зробити самостійно.

>>> list_1 = ['Ace', 'King']
>>> list_2 = ['Spades', 'Clubs', 'Diamonds']
>>> deck = []
>>> for i in range(max((len(list_1),len(list_2)))):
        while True:
            try:
                card = (list_1[i],list_2[i])
            except IndexError:
                if len(list_1)>len(list_2):
                    list_2.append('')
                    card = (list_1[i],list_2[i])
                elif len(list_1)<len(list_2):
                    list_1.append('')
                    card = (list_1[i], list_2[i])
                continue
            deck.append(card)
            break
>>>
>>> #and the result should be:
>>> print deck
>>> [('Ace', 'Spades'), ('King', 'Clubs'), ('', 'Diamonds')]

2
Зміна одного із вхідних списків (якщо вони різняться за довжиною) не є приємним побічним ефектом. Крім того, два завдання cardв програмі if-elifне потрібні, тому у вас є continue. (Насправді, без continueвас не було б змінити списки: як раніше згадані завдання повинні бути потім збережені і стали card = (list_1[i], '')і card = ('', list_2[1])відповідно.)
ᴠɪɴᴄᴇɴᴛ

5

Вихід, який ви показали в постановці проблеми, - це не кортеж, а список

list_c = [(1,5), (2,6), (3,7), (4,8)]

перевірити для

type(list_c)

вважаючи, що ви хочете, щоб результат вийшов із списку list_a та list_b, зробіть

tuple(zip(list_a,list_b)) 

З моєї точки зору, мені здається, що я шукаю і добре працюю як для списку, так і для кортежу. Тому що, використовуючи print , ви побачите потрібне значення (як очікували та згадували @cyborg та @Lodewijk) і нічого, що стосується об'єкта, такого як: <map object at 0x000001F266DCE5C0>або <zip object at 0x000002629D204C88>. Принаймні, рішення щодо карти та zip (поодинці) здається мені неповним (або занадто складним).
Wagner_SOFC

1
Питання стверджує, що вони хочуть список кортежів, а не кортеж кортежів.
goryh

1

Одна альтернатива без використання zip:

list_c = [(p1, p2) for idx1, p1 in enumerate(list_a) for idx2, p2 in enumerate(list_b) if idx1==idx2]

Якщо ви хочете отримати не лише кортежі 1-го, 1-го, 2-го з 2-го…, але й усі можливі комбінації двох списків, це було б зроблено з

list_d = [(p1, p2) for p1 in list_a for p2 in list_b]
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.