Wydrukować listę w odwrotnej kolejności za pomocą range ()?

364

Jak możesz utworzyć następującą listę range()w Pythonie?

[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
ramesh.mimit
źródło
Co z [* zakresem (9, -1, -1)]? Super pytoniczny!
onofricamila

Odpowiedzi:

561

użyj reversed()funkcji:

reversed(range(10))

To jest o wiele bardziej znaczące.

Aktualizacja:

Jeśli chcesz, aby była to lista (jak wskazał btk):

list(reversed(range(10)))

Aktualizacja:

Jeśli chcesz użyć tylko rangedo osiągnięcia tego samego rezultatu, możesz użyć wszystkich jego parametrów.range(start, stop, step)

Na przykład, aby wygenerować listę [5,4,3,2,1,0], możesz użyć:

range(5, -1, -1)

Może to być mniej intuicyjne, ale jak wspomniano w komentarzach, jest to bardziej wydajne i właściwe użycie zakresu dla odwróconej listy.

Michał Šrajer
źródło
13
Chociaż „jest” mniej wydajny. I nie można na nim wykonywać operacji krojenia.
Jakob Bowyer,
6
Ta odpowiedź jest bardzo jasna i łatwa do zrozumienia, ale musi być range(10), nie range(9). Ponadto, jeśli chcesz mieć w pełni uformowaną listę (do krojenia itp.), Powinieneś to zrobić list(reversed(range(10))).
John Y
5
W ten sposób powstaje iterator, OP poprosił o wynik w postaci listy.
btk
6
Używanie „odwróconego” z generatorem Pythona (zakładając, że mówimy o wbudowanym zakresie Pythona 3) jest po prostu błędne koncepcyjnie i uczy złych nawyków nieuwzględniania złożoności pamięci / przetwarzania podczas programowania w języku wysokiego poziomu.
thedk
7
W Python3 ogólnie reversednie akceptuje generatorów, ale akceptuje range. Dlaczego reversed(range(10000))trzeba przydzielić pamięć dla całej listy? rangemoże zwrócić obiekt, który implementuje __reversed__metodę, która umożliwia wydajną iterację wsteczną?
avmohan,
312

Skorzystaj z wbudowanej funkcji „range”. Podpis jest range(start, stop, step). W ten sposób powstaje sekwencja, która daje liczby, zaczynając od starti kończąc, jeśli stopzostała osiągnięta, z wyłączeniem stop.

>>> range(9,-1,-1)   
    [9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
>>> range(-2, 6, 2)
    [-2, 0, 2, 4]

W Pythonie 3 tworzy to obiekt nie będący listą range, który działa skutecznie jak lista tylko do odczytu (ale zużywa znacznie mniej pamięci, szczególnie w przypadku dużych zakresów).

Pan B.
źródło
1
Czy możesz to również wyjaśnić, czy możesz mi polecić dowolną stronę internetową / książkę pdf dla python
ramesh.mimit
8
@ramesh Jeśli uruchomisz help(range)w powłoce pytona, poda ci argumenty. Są to liczba na początek, liczba na końcu (wyłącznie) i krok do podjęcia, więc zaczyna się od 9 i odejmuje 1, aż dojdzie do -1 (w tym momencie zatrzymuje się bez powrotu, dlatego zakres kończy się o 0)
Michael Mrozek
3
@ ramesh.mimit: Przejdź na oficjalną stronę Pythona. Jest tam pełna dokumentacja, w tym świetny samouczek.
John Y,
5
Naprawdę jest to właściwa odpowiedź, ale można ją dokładniej wyjaśnić. range(start, stop, step) - zacznij od liczby starti uzyskaj wyniki, o ile stopnie zostały osiągnięte, przesuwając się za stepkażdym razem.
Pan B
43

Możesz użyć range(10)[::-1]tej samej rzeczy range(9, -1, -1)i prawdopodobnie bardziej czytelnej (jeśli znasz zwykły sequence[::-1]idiom Pythona).

martineau
źródło
28

Dla tych, którzy są zainteresowani „wydajnością” dotychczas zebranych opcji ...

Odpowiedź Jaime'a RGP skłoniła mnie do ponownego uruchomienia komputera po zorientowaniu się w czasie nieco „trudnego” rozwiązania Jasona dosłownie po mojej własnej sugestii (poprzez komentarz). Aby zaoszczędzić ciekawskim przestojów, przedstawiam tutaj moje wyniki (najgorsze pierwsze):

Odpowiedź Jasona (może po prostu wycieczka w moc rozumienia listy ):

$ python -m timeit "[9-i for i in range(10)]"
1000000 loops, best of 3: 1.54 usec per loop

odpowiedź martineau (czytelna, jeśli znasz składnię rozszerzonych plasterków ):

$ python -m timeit "range(10)[::-1]"
1000000 loops, best of 3: 0.743 usec per loop

Odpowiedź Michała Šrajera (przyjęta, bardzo czytelna):

$ python -m timeit "reversed(range(10))"
1000000 loops, best of 3: 0.538 usec per loop

odpowiedź Benego (pierwsza, ale bardzo szkicowa w tym czasie ):

$ python -m timeit "range(9,-1,-1)"
1000000 loops, best of 3: 0.401 usec per loop

Ostatnia opcja jest łatwa do zapamiętania za pomocą range(n-1,-1,-1)notacji Val Neekmana .

Wilk
źródło
3
Przed przeczytaniem tej odpowiedzi wykonałem własne pomiary i uzyskałem te same wyniki.
Todd Owen,
13
for i in range(8, 0, -1)

rozwiąże ten problem. Wyjdzie 8 do 1, a -1 oznacza odwróconą listę

Jutta
źródło
10

Nie ma sensu używać, reverseponieważ metoda zakresu może zwrócić listę odwróconą.

Gdy masz iterację n elementów i chcesz zastąpić zwróconą przez range(start, stop, step)ciebie kolejność listy , musisz użyć trzeciego parametru zakresu, który identyfikuje stepi ustawia go -1, pozostałe parametry należy odpowiednio dostosować:

  1. Zapewnić zatrzymanie parametr jak -1(to poprzednia wartość stop - 1, stopbyła równa 0).
  2. Jako parametr początkowy użyj n-1.

Odpowiednikiem zakresu (n) w odwrotnej kolejności byłoby:

n = 10
print range(n-1,-1,-1) 
#[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
Andrij Ivaneyko
źródło
6
Szczerze mówiąc, uważam to za mniej intuicyjne niżreversed(range(n))
Nicholas Hamilton
1
@NicholasHamilton Zgadzam się z tobą, jednak pomysł na tę odpowiedź polega na zużyciu mniejszych zasobów ... (a pytanie dotyczyło użycia tylko funkcji zakresu :))
Andriy Ivaneyko
Ok, nie martw się, myślę, że to zależy od sytuacji. Na przykład, jeśli funkcja zakresu jest używana jako indeks dla pętli, będzie to miało ograniczony efekt, jednak jeśli zostanie użyta w pętli zewnętrznej, wówczas koszt się zwiększy ...
Nicholas Hamilton,
8

Poza tym czytelność reversed(range(n))wydaje się szybsza niż range(n)[::-1].

$ python -m timeit "reversed(range(1000000000))"
1000000 loops, best of 3: 0.598 usec per loop
$ python -m timeit "range(1000000000)[::-1]"
1000000 loops, best of 3: 0.945 usec per loop

Tylko jeśli ktoś się zastanawiał :)

Jaime RGP
źródło
dobrze mieć jakieś numery :) - dlaczego też nie dodałeś range(1000000000-1,-1,-1)?
Wolf
... lepiej nie próbuj timeit range(1000000000-1,-1,-1)w wierszu poleceń, zamiast tego zobacz moje wyniki :-)
Wolf
6

Wymaganie zawarte w tym pytaniu wymaga listliczby całkowitej 10 w porządku malejącym. Stwórzmy więc listę w Pythonie.

# This meets the requirement.
# But it is a bit harder to wrap one's head around this. right?
>>> range(10-1, -1, -1)
[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]

# let's find something that is a bit more self-explanatory. Sounds good?
# ----------------------------------------------------

# This returns a list in ascending order.
# Opposite of what the requirement called for.
>>> range(10)
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

# This returns an iterator in descending order.
# Doesn't meet the requirement as it is not a list.
>>> reversed(range(10))
<listreverseiterator object at 0x10e14e090>

# This returns a list in descending order and meets the requirement
>>> list(reversed(range(10)))
[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
un33k
źródło
2
Bardzo podoba mi się potrójny-1 To właśnie ten wzór ułatwia utrzymanie wrap one's head around this- Dzisiaj dowiedziałem się, że jeśli naprawdę musi być wydajny , używaj go, range(n-1, -1, -1)gdy przemierzasz zasięg w odwrotnej kolejności.
Wolf
5

Możesz drukować liczby odwrotne za pomocą Range () BIF Like,

for number in range ( 10 , 0 , -1 ) :
    print ( number ) 

Wynik wyniesie [10,9,8,7,6,5,4,3,2,1]

range () - zakres (początek, koniec, przyrost / spadek), gdzie początek obejmuje, koniec jest wyłączny, a przyrost może być dowolnymi liczbami i zachowuje się jak krok

Pan Suryaa Jha
źródło
5

Często zadawane pytanie brzmi, czy range(9, -1, -1)lepiej niż reversed(range(10))w Pythonie 3? Ludzie, którzy pracowali w iteratorach w innych językach, od razu myślą, że reverse () musi buforować wszystkie wartości, a następnie wracać w odwrotnej kolejności. Chodzi o to, że reversed()operator Pythona nie działa, jeśli obiekt jest tylko iteratorem. Obiekt musi mieć jeden z dwóch poniżej dwóch, aby funkcja reverse () działała:

  1. len()Indeksy wsparcia i całkowite za pośrednictwem[]
  2. Lub __reversed__()zaimplementuj metodę.

Jeśli spróbujesz użyć odwróconej () na obiekcie, który nie ma żadnego z powyższych, otrzymasz:

>>> [reversed((x for x in range(10)))]
TypeError: 'generator' object is not reversible

Krótko mówiąc, Python reversed()jest przeznaczony tylko do obiektów podobnych do tablicy, więc powinien mieć taką samą wydajność jak iteracja do przodu.

Ale co z range()? Czy to nie generator? W Pythonie 3 jest to generator, ale zawinięty w klasę, która implementuje oba powyższe. Więc range(100000)nie zajmuje dużo pamięci, ale nadal obsługuje wydajne indeksowanie i cofanie.

Podsumowując, możesz używać reversed(range(10))bez żadnego wpływu na wydajność.

Shital Shah
źródło
3

wierzę, że to może pomóc,

range(5)[::-1]

poniżej jest użycie:

for i in range(5)[::-1]:
    print i 
dineshsprabu
źródło
3
range(9,-1,-1)
[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
Asinox
źródło
13
Twoja odpowiedź pokazuje, dlaczego reversed(range(10))jest mniej podatna na błędy. Bez obrazy Asinox. Tylko uczciwa obserwacja.
Michał Šrajer
Nie wiem, czy standardem jest pozostawienie błędnych odpowiedzi na wyświetlaczu. Czy ja lub ktoś mogę to poprawić, a nawet usunąć?
sinekonata
3
@sine, nie, po prostu zostaw to i zastanawiaj się, jak to nagromadziło 3 głosy poparcia ... Przypuszczam, że mógłbyś oflagować go dla uwagi moderatora (duplikat odpowiedzi z 18 miesięcy wcześniej, z wyjątkiem zepsutych), nie jestem pewien, czy go usuną.
OGHaza
Dla wszystkich przyszłych odniesień: tylko kod nie jest VLQ , a błędne odpowiedzi również nie powinny być oflagowane . Oddaj głos i przejdź dalej, ale w tym przypadku oflagowanie jest niewłaściwe, bez względu na to, czy jest to NAA czy mod-flagowanie. - Z recenzji
Zoe
2

Używanie bez [:: - 1] lub odwróconego -

def reverse(text):
    result = []
    for index in range(len(text)-1,-1,-1):
        c = text[index]
        result.append(c)
    return ''.join(result)

print reverse("python!")
rhel.user
źródło
6
Dlaczego ktoś miałby to pisać zamiast "python!"[::-1]?
Daniel
1
[9-i for i in range(10)]
[9, 8, 7, 6, 5, 4, 3, 2, 1, 0]
Jason
źródło
1

Nie musisz koniecznie używać funkcji zakresu, możesz po prostu zrobić listę [:: - 1], która powinna szybko zwrócić listę w odwrotnej kolejności, bez żadnych dodatków.

użytkownik7420740
źródło
To wydaje się być rozwiązaniem sugerowanym w poprzedniej odpowiedzi . Wygląda też na to, że próbujesz skomentować inną poprzednią odpowiedź - zanim będziesz mógł to zrobić, musisz uzyskać trochę więcej reputacji.
Grisha Levit
1

Załóżmy, że masz listę o nazwie a = {1,2,3,4,5} Teraz, jeśli chcesz wydrukować listę w odwrotnej kolejności, po prostu użyj następującego kodu.

a.reverse
for i in a:
   print(i)

Wiem, że zapytałeś za pomocą zasięgu, ale już na nie odpowiedział.

Vishal Kumar
źródło
0
range(9,-1,-1)
    [9, 8, 7, 6, 5, 4, 3, 2, 1, 0]

Jest poprawna forma. Jeśli użyjesz

reversed(range(10))

nie dostaniesz 0 przypadku. Powiedzmy na przykład, że twoja 10 nie jest liczbą magiczną, a zmienna, której używasz do wyszukiwania, zaczyna się od tyłu. Jeśli twoja n wynosi 0, odwrócony (zakres (0)) nie zostanie wykonany, co jest błędne, jeśli przypadkiem masz pojedynczy obiekt w indeksie zerowym.

użytkownik3807372
źródło
0

Pomyślałem, że wielu (jak ja) może być bardziej zainteresowanych zwykłym przypadkiem przeszukiwania istniejącej listy w odwrotnej kolejności , jak to podano w tytule, zamiast generowania indeksów dla takiego przejścia.

Pomimo tego, że wszystkie prawidłowe odpowiedzi są w tym przypadku idealnie w porządku, chcę zauważyć, że porównanie wydajności wykonane w odpowiedzi Wolfa służy wyłącznie do generowania indeksów. Podjąłem więc podobny test porównawczy dla istniejącej listy w odwrotnej kolejności.

TL; DR a[::-1] jest najszybszy.

Wymagania wstępne:

a = list(range(10))

Odpowiedź Jasona :

%timeit [a[9-i] for i in range(10)]
1.27 µs ± 61.5 ns per loop (mean ± std. dev. of 7 runs, 1000000 loops each)

odpowiedź martineau :

%timeit a[::-1]
135 ns ± 4.07 ns per loop (mean ± std. dev. of 7 runs, 10000000 loops each)

Odpowiedź Michała Šrajera :

%timeit list(reversed(a))
374 ns ± 9.87 ns per loop (mean ± std. dev. of 7 runs, 1000000 loops each)

odpowiedź Bene'a :

%timeit [a[i] for i in range(9, -1, -1)]
1.09 µs ± 11.1 ns per loop (mean ± std. dev. of 7 runs, 1000000 loops each)

Jak widać, w tym przypadku nie ma potrzeby jawnego generowania indeksów, więc najszybszą metodą jest ta, która wykonuje mniej dodatkowych działań.

NB: Testowałem w JupyterLab, który ma przydatne „magiczne polecenie” %timeit. Używa standardu timeit.timeitpod maską. Przetestowano pod kątem Python 3.7.3

pkuderov
źródło