pytutorial/python_basics/for/README.md
David Rotermund 656bfba7a5
Update README.md
Signed-off-by: David Rotermund <54365609+davrot@users.noreply.github.com>
2023-12-07 17:48:04 +01:00

4.5 KiB
Raw Blame History

Flow Control for-loop

{:.no_toc}

* TOC {:toc}

The goal

For what reason...

Questions to David Rotermund

Logic blocks need to be indented. Preferable with 4 spaces!

The for statement

The for statement is used to iterate over the elements of a sequence (such as a string, tuple or list) or other iterable object

With range()

for i in range(0, 3):
    print(i)

Output:

0
1
2

range()

class range(stop)
class range(start, stop[, step])

With a list

for i in [0, "A", 7, "nom num"]:
    print(i)

Output:

0
A
7
nom num

The full statement

for_stmt ::=  "for" target_list "in" starred_list ":" suite
              ["else" ":" suite]

for loop (the truth)

{: .topic-optional} This is an optional topic!

The for loop is not counting up itself. It uses the __iter__ & __next__ combo of a instance of a class or a generator via yield.

Class

class ClassIterExample:
    counter: int
    counter_max: int

    def __init__(self):
        self.counter = 8
        self.counter_max = 12

    def __iter__(self):
        return self

    def __next__(self):
        if self.counter < self.counter_max:
            self.counter += 1
        else:
            raise StopIteration

        important_result: int = self.counter**2

        return f"Element: {self.counter}", important_result


test_instance = ClassIterExample()

for i in test_instance:
    print(i)

Output:

('Element: 9', 81)
('Element: 10', 100)
('Element: 11', 121)
('Element: 12', 144)

Generator yield

from typing import Generator


def do_more(min_value: int, max_value: int) -> Generator:
    for index in range(min_value, max_value):
        yield index**2


for i in do_more(min_value=3, max_value=8):
    print(i)

Output:

9
16
25
36
49

for loop -- enumerate

enumerate(iterable, start=0)

If you need the index of the element then you can use enumerate:

for i in enumerate(range(10, 13)):
    print(i)

Output:

(0, 10)
(1, 11)
(2, 12)

{: .topic-optional} This is an optional topic!

for i in enumerate(range(10, 13)):
    print(i)
else:
    print("*")
    print(i)

Output:

(0, 10)
(1, 11)
(2, 12)
*
(2, 12)

{: .topic-optional} This is an optional topic!

class ClassIterExample:
    counter: int
    counter_max: int

    def __init__(self):
        self.counter = 8
        self.counter_max = 12

    def __iter__(self):
        return self

    def __next__(self):
        if self.counter < self.counter_max:
            self.counter += 1
        else:
            raise StopIteration

        important_result: int = self.counter**2

        return f"Element: {self.counter}", important_result


test_instance = ClassIterExample()

for i in enumerate(test_instance):
    print(i)

Output:

(0, ('Element: 9', 81))
(1, ('Element: 10', 100))
(2, ('Element: 11', 121))
(3, ('Element: 12', 144))

You see a tuple with the enumerate id as a first component and a second tuple as result from the iterator.

break

break may only occur syntactically nested in a for or while loop, but not nested in a function or class definition within that loop.

It terminates the nearest enclosing loop, skipping the optional else clause if the loop has one.

If a for loop is terminated by break, the loop control target keeps its current value.

break_stmt ::=  "break"
for i in range(0, 5):
    if i == 2:
        break
    print(i)

Output:

0
1

continue

continue may only occur syntactically nested in a for or while loop, but not nested in a function or class definition within that loop. It continues with the next cycle of the nearest enclosing loop.

continue_stmt ::=  "continue"
for i in range(0, 5):
    if i == 2:
        continue
    print(i)

Output:

0
1
3
4