W większości przypadków łatwiej (i taniej) jest zrobić pierwszą iterację specjalnym przypadkiem zamiast ostatniego:
first = True
for data in data_list:
if first:
first = False
else:
between_items()
item()
Będzie to działać dla każdego iterowalnego, nawet dla tych, które nie mają len()
:
file = open('/path/to/file')
for line in file:
process_line(line)
# No way of telling if this is the last line!
Poza tym nie sądzę, że istnieje ogólnie lepsze rozwiązanie, ponieważ zależy to od tego, co próbujesz zrobić. Na przykład, jeśli budujesz ciąg z listy, naturalnie lepiej jest użyć str.join()
niż for
pętli „ze specjalnym przypadkiem”.
Stosując tę samą zasadę, ale bardziej kompaktową:
for i, line in enumerate(data_list):
if i > 0:
between_items()
item()
Wygląda znajomo, prawda? :)
W przypadku @ofko i innych, którzy naprawdę muszą dowiedzieć się, czy aktualna wartość iterowalności bez len()
jest ostatnia, musisz spojrzeć w przyszłość:
def lookahead(iterable):
"""Pass through all values from the given iterable, augmented by the
information if there are more values to come after the current one
(True), or if it is the last value (False).
"""
# Get an iterator and pull the first value.
it = iter(iterable)
last = next(it)
# Run the iterator to exhaustion (starting from the second value).
for val in it:
# Report the *previous* value (more to come).
yield last, True
last = val
# Report the last value.
yield last, False
Następnie możesz użyć tego w następujący sposób:
>>> for i, has_more in lookahead(range(3)):
... print(i, has_more)
0 True
1 True
2 False