2013-05-01 14 views
6

mam ten aktualnej listy ze zrozumieniem:Python - Lista zrozumienie z wielu argumentów w za

... 
cur = [[14, k, j] for j, k in rows[14], range(15)] 
... 

i to daje mi następujący błąd:

...   
    cur = [[14, k, j] for j, k in rows[14], range(15)] 
ValueError: too many values to unpack 

Każda pomoc docenione w jaki sposób naprawiłbym to. Po prostu nie chcę napisać pełnej pętli for lub całej listy ręcznie. Dziękuję Ci! : D

Dodatkowe info:

rows = [{1: '75'}, 
     {1: '95', 2: '64'}, 
     {1: '17', 2: '47', 3: '82'}, 
     {1: '18', 2: '35', 3: '87', 4: '10'}, 
     {1: '20', 2: '04', 3: '82', 4: '47', 5: '65'}, 
     {1: '19', 2: '01', 3: '23', 4: '75', 5: '03', 6: '34'}, 
     {1: '88', 2: '02', 3: '77', 4: '73', 5: '07', 6: '63', 7: '67'}, 
     {1: '99', 2: '65', 3: '04', 4: '28', 5: '06', 6: '16', 7: '70', 8: '92'}, 
     {1: '41', 2: '41', 3: '26', 4: '56', 5: '83', 6: '40', 7: '80', 8: '70', 9: '33'}, 
     {1: '41', 2: '48', 3: '72', 4: '33', 5: '47', 6: '32', 7: '37', 8: '16', 9: '94', 10: '29'}, 
     {1: '53', 2: '71', 3: '44', 4: '65', 5: '25', 6: '43', 7: '91', 8: '52', 9: '97', 10: '51', 11: '14'}, 
     {1: '70', 2: '11', 3: '33', 4: '28', 5: '77', 6: '73', 7: '17', 8: '78', 9: '39', 10: '68', 11: '17', 12: '57'}, 
     {1: '91', 2: '71', 3: '52', 4: '38', 5: '17', 6: '14', 7: '91', 8: '43', 9: '58', 10: '50', 11: '27', 12: '29', 13: '48'}, 
     {1: '63', 2: '66', 3: '04', 4: '68', 5: '89', 6: '53', 7: '67', 8: '30', 9: '73', 10: '16', 11: '69', 12: '87', 13: '40', 14: '31'}, 
     {1: '04', 2: '62', 3: '98', 4: '27', 5: '23', 6: '09', 7: '70', 8: '98', 9: '73', 10: '93', 11: '38', 12: '53', 13: '60', 14: '04', 15: '23'}] 

Odpowiedz

7

Trzeba je zip iteracyjne tak:

cur = [[14, k, j] for j, k in zip(rows[14], range(15))] 
+0

Czym dokładnie jest skompresowanie? Z czystej ciekawości. –

+1

@EthanBrouwer Połączyłem go, w zasadzie można tylko iterować przez pojedynczą iterowalną w pętli 'for', aby iterować dwie na raz, musisz użyć' zip' który tworzy iterable, który zwraca krotkę odpowiednich elementów na każdy indeks z list. – jamylak

+0

Czy to oznacza, że ​​muszę też (j, k) czy to nie ma znaczenia? –

1

Rozszerzając użytkownika @ jamylak odpowiedź: Alternatywnie można użyć map

cur = [[14, k, j] for j, k in map(None,rows[14], range(15))] 

Spowoduje to wyświetlenie krótszych list ne.

+2

'itertools.izip_longest' to lepszy sposób na zrobienie tego – jamylak

+0

@mgilson o nie! edytuj: poczekaj, nie ma, tylko 'Brak' rzeczy – jamylak

+1

@jamylak - Korekta. Wywołanie 'map' z' None' znika w python3.x – mgilson

4

Aby wyjaśnić swój kod:

cur = [[14, k, j] for j, k in rows[14], range(15)] 

jest taka sama, jak:

cur = [[14, k, j] for j, k in (rows[14], range(15))] 

Teraz widzimy wyraźniej, że stworzyliśmy tuple i iteracji nad nim. Po raz pierwszy w pętli krotka rezygnuje z rows[14], która jest słownikiem zawierającym więcej niż 2 elementy, więc nie można jej rozpakować do j i k.

Jak zauważył jamylak, kluczem jest zip dwóch iterables razem.

cur = [[14, k, j] for j,k in zip(rows[14],range(15))] 

Można myśleć o nim jak zamek błyskawiczny:

zip(a,b) = [ 
    (a[0], b[0]), 
    (a[1], b[1]), 
    (a[2], b[2]), 
    ... 
    } 

rozpisane w ten sposób można zobaczyć, w jaki sposób struktura rodzaj przypomina zamek (z a i b jest lewy i prawy kawałki Po skompresowaniu pasujesz do elementu po lewej stronie z elementem po prawej stronie. Oczywiście obiekty, które przechodzą na zip, nie muszą być indeksowalne (ważne jest tylko to, że możesz je powtórzyć) i możesz "zip" więcej niż 2 iterables razem ...

+0

'rows [14]' to ostatni 'słownik' w wierszach' lista' –

+0

@EthanBrouwer - Dzięki. To pomaga (chyba powinienem był przeczytać dokładniej) – mgilson