목록의 요소 쌍 결합


82

목록을 결합하여 다음과 같이 하나의 긴 문자열을 만들 수 있음을 알고 있습니다.

x = ['a', 'b', 'c', 'd']
print ''.join(x)

분명히 이것은 다음을 출력합니다.

'abcd'

그러나 내가하려는 것은 목록의 첫 번째와 두 번째 문자열을 결합한 다음 세 번째와 네 번째 문자열을 결합하는 것입니다. 요컨대, 위의 예에서 대신 다음과 같은 출력을 얻습니다.

['ab', 'cd']

이 작업을 수행하는 간단한 방법이 있습니까? 문자열의 수는 항상 짝수이지만 목록 내의 문자열 수와 마찬가지로 목록의 문자열 길이도 예측할 수 없다는 점도 언급해야합니다. 따라서 원래 목록은 다음과 같을 수 있습니다.

['abcd', 'e', 'fg', 'hijklmn', 'opq', 'r'] 

“목록에있는 문자열의 길이를 예측할 수 없다는 점도 언급해야합니다.” – 길이가 중요합니까? 즉, 모든 목록 요소 쌍을 결합하고 싶습니까, 아니면 결과 요소가 특정 길이 제한 아래로 유지되는 한 실제로 콘텐츠를보고 결합하고 싶습니까?
찌를

단순히 모든 페어에 가입하십시오. 페어 수를 모르는 것이 문제가 될 수 있다고 생각했습니다
John

답변:


75

단계에 슬라이스 표기법을 사용할 수 있습니다.

>>> x = "abcdefghijklm"
>>> x[0::2] #0. 2. 4...
'acegikm'
>>> x[1::2] #1. 3. 5 ..
'bdfhjl'
>>> [i+j for i,j in zip(x[::2], x[1::2])] # zip makes (0,1),(2,3) ...
['ab', 'cd', 'ef', 'gh', 'ij', 'kl']

목록에도 동일한 논리가 적용됩니다. 단순히 두 개의 문자열을 함께 추가하기 때문에 문자열 길이는 중요하지 않습니다.


1
kevpie의 대답이 훨씬 낫다는 것은 의심의 여지가 없습니다. 여기서는 x[:::2]객체를 x[1::2]생성하고 다른 객체를 생성합니다. 이러한 생성은 아마도 내부의 인덱스 계산을 기반으로 할 것입니다.이 두 객체가 인수로 전달 된 함수에 대한 호출은 연속적인 요소 쌍을 얻기 전에 필요합니다. 연결되어야합니다. kevpie의 대답에는 하나의 반복자가 생성되고 반복은 인덱스를 처리하지 않고도 손대지 않은 목록의 요소에서 요소로 건너 뛰며 훨씬 더 비단뱀입니다.
eyquem 2014

itertools.islice[] 대신 사용 하는 @eyquem 은 중간 개체를 제거합니다. 그러나 두 답변 모두 동일한 조건에서 작동하고 동일하게 반환되기 때문에 둘 다 맞습니다. 그리고 zip(i[::2], i[1::2])나에게 너무 달콤 해 보이는데, 왜 안돼? :)
utdemir

이것은 시퀀스 에서만 작동 하지만 @kevpie의 대답은 더 일반적이며 반복 가능한 모든 작업에서 작동합니다 .
Kos

37

반복자를 사용하십시오.

목록 이해력 :

>>> si = iter(['abcd', 'e', 'fg', 'hijklmn', 'opq', 'r'])
>>> [c+next(si, '') for c in si]
['abcde', 'fghijklmn', 'opqr']
  • 메모리 사용에 매우 효율적입니다.
  • s의 정확히 하나의 순회

생성기 표현 :

>>> si = iter(['abcd', 'e', 'fg', 'hijklmn', 'opq', 'r'])
>>> pair_iter = (c+next(si, '') for c in si)
>>> pair_iter # can be used in a for loop
<generator object at 0x4ccaa8>
>>> list(pair_iter) 
['abcde', 'fghijklmn', 'opqr']
  • 반복자로 사용

map, str .__ add__, iter 사용

>>> si = iter(['abcd', 'e', 'fg', 'hijklmn', 'opq', 'r'])
>>> map(str.__add__, si, si)
['abcde', 'fghijklmn', 'opqr']

next (iterator [, default]) 는 Python 2.6부터 사용할 수 있습니다.


2
단연코 최고의 답변입니다. utdemir의 답변에 대한 내 의견을 참조하십시오.
eyquem 2014

4

비단뱀처럼 :-)

>>> x = ['a1sd','23df','aaa','ccc','rrrr', 'ssss', 'e', '']
>>> [x[i] + x[i+1] for i in range(0,len(x),2)]
['a1sd23df', 'aaaccc', 'rrrrssss', 'e']

목록 길이가 이상하면 놀라고 싶다면 다음을 시도하십시오.

[x[i] + x[i+1] if not len(x) %2 else 'odd index' for i in range(0,len(x),2)]

행운을 빌어 요


2

임시 목록을 작성하지 않고 :

>>> import itertools
>>> s = 'abcdefgh'
>>> si = iter(s)
>>> [''.join(each) for each in itertools.izip(si, si)]
['ab', 'cd', 'ef', 'gh']

또는:

>>> import itertools
>>> s = 'abcdefgh'
>>> si = iter(s)
>>> map(''.join, itertools.izip(si, si))
['ab', 'cd', 'ef', 'gh']

좋지만 내 코드를 고려할 때 어쨌든 원래 목록부터 시작하는 것을 고려하면 utdmr을 선택하는 것이 좋지 않다고 생각합니다 .... 감사합니다
John

1
>>> lst =  ['abcd', 'e', 'fg', 'hijklmn', 'opq', 'r'] 
>>> print [lst[2*i]+lst[2*i+1] for i in range(len(lst)/2)]
['abcde', 'fghijklmn', 'opqr']

1

제가 Regs를 잘 못하기 때문에 이런 식으로 할 것입니다 ..

암호

t = '1. eat, food\n\
7am\n\
2. brush, teeth\n\
8am\n\
3. crack, eggs\n\
1pm'.splitlines()

print [i+j for i,j in zip(t[::2],t[1::2])]

산출:

['1. eat, food   7am', '2. brush, teeth   8am', '3. crack, eggs   1pm']  

도움이 되었기를 바랍니다 :)

당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.