술어와 일치하는 목록에서 첫 번째 요소를 찾는 관용적 방법을 원합니다.
현재 코드는 매우 추악합니다.
[x for x in seq if predicate(x)][0]
나는 그것을 다음과 같이 바꾸는 것에 대해 생각했다.
from itertools import dropwhile
dropwhile(lambda x: not predicate(x), seq).next()
그러나 더 우아한 것이 있어야합니다 … 그리고 None
일치하는 것이 없으면 예외를 제기하는 대신 값을 반환하면 좋을 것 입니다.
나는 다음과 같은 함수를 정의 할 수 있다는 것을 알고있다.
def get_first(predicate, seq):
for i in seq:
if predicate(i): return i
return None
그러나 이미 동일한 기능을 제공하는 내장 기능이있는 경우 이와 같은 유틸리티 기능으로 코드를 채우기 시작하는 것은 무의미합니다.
답변
순서 seq
와 일치하는 첫 번째 요소를 찾으려면 predicate
:
next(x for x in seq if predicate(x))
또는 ( itertools.ifilter
Python 2에서) :
next(filter(predicate, seq))
그것은 제기 StopIteration
존재하지 않는 경우.
None
그러한 요소가없는 경우 리턴하려면 다음을 수행하십시오.
next((x for x in seq if predicate(x)), None)
또는:
next(filter(predicate, seq), None)
답변
기본값으로 생성기 표현식을 사용한 후 다음을 사용할 next
수 있습니다.
next((x for x in seq if predicate(x)), None)
이 one-liner의 경우 Python> = 2.6을 사용해야합니다.
이 인기있는 기사는이 문제에 대해 더 자세히 설명합니다. 가장 깨끗한 Python 찾기 기능? .
답변
귀하의 질문에 제안한 솔루션에 문제가 있다고 생각하지 않습니다.
내 자신의 코드에서는 다음과 같이 구현합니다.
(x for x in seq if predicate(x)).next()
with 구문은을 ()
사용하여 모든 목록을 한 번에 생성하는 것보다 효율적인 생성기 를 만듭니다 []
.
답변
JF Sebastian의 대답은 가장 우아하지만 포트란이 지적한 것처럼 Python 2.6이 필요합니다.
파이썬 버전 <2.6의 경우, 내가 생각해 낼 수있는 최선의 방법은 다음과 같습니다.
from itertools import repeat,ifilter,chain
chain(ifilter(predicate,seq),repeat(None)).next()
또는 나중에 목록이 필요하거나 (목록이 StopIteration을 처리 함) 첫 번째 항목 이상을 필요로하는 경우에도 islice를 사용하여 수행 할 수 있습니다.
from itertools import islice,ifilter
list(islice(ifilter(predicate,seq),1))
업데이트 : 개인적으로 StopIteration을 포착하고 None을 반환하는 first ()라는 미리 정의 된 함수를 사용하고 있지만 위의 예에 비해 가능한 개선 사항은 다음과 같습니다. filter / ifilter를 사용하지 마십시오.
from itertools import islice,chain
chain((x for x in seq if predicate(x)),repeat(None)).next()