Direkt zum Hauptbereich

Python : Generatoren

Einführung

In diesem kurzen Blogartikel beschäftigen wir uns mit Funktions-Dekoratoren (function decorators) und Generatoren (generators). Ein Dekorator ist eine Funktion, die eine andere Funktion als Parameter akzeptiert und eine sogenannte innere Funktion zurückgibt. Dekoratoren sind nützlich, um zusätzliche Verarbeitungsschritte für eine Funktion durchzuführen. Ein Generator hingegen ist eine Funktion die Werte sequentiell generiert. Anstatt alle Werte auf einmal zurückzugeben, wird der Generator bei seinem nächsten Aufruf den nächsten anfallenden Wert in der Sequenz zurückgeben. 

Function Decorators

Oftmals sind viele Funktionen schon vorhanden, doch nicht konkret für einen Zweck verwendbar. Es sind einige Arbeitsschritte zusätzlich nötig, um mit der vorhandenen Funktion das gewünschte Resultat zu erzielen. Anstatt die vorhandene Funktion neu zu implementieren, wird die bestehende eigentlich nützliche Funktion durch eine andere Funktion dekoriert. Sie wird also verwendet, um mit dem Dekorator ein modifiziertes, zweckmässigeres Resultat zu erzielen:


def decor(fun):
    def inner(arg):
        value = fun(arg)
        return value * 2
    return inner


mysum = decor(sum)

print(mysum([1, 2, 3, 4]))
# 20
Im obigen Beispiel wurde die Funktion sum der Standardbibliothek dekoriert. Sie gibt die Summe aller Listenelemente zurück. Der Dekorator hat zum Zweck die doppelte Summe zurückzugeben. Mit @ Annotationen wird die Sache noch einfacher. Man setzt einfach vor die zu dekorierende Funktion ein @dekorname:

@decor
def num():
    return 5

print(num())
# Hier wird nun automatisch die dekorierte Funktion genommen 

Dabei kann eine Funktion auch mehrfach dekoriert werden.

@dec1
@dec2
def fun(arg1, arg2, …):
    pass
# aequivalent zu

def fun(arg1, arg2, …):
    pass

func = dec1(dec2(fun))
 

Generatoren

Generatoren sind Funktionen, die eine Wertesequenz zurückgeben über die iteriert werden kann. Sie sind gewöhnliche Funktionen, enthalten aber das Schlüsselwort yield. Mit yield wird einerseits der Funktion mitgeteilt, dass sie nun pausieren soll und andererseits, dass sie einen Wert an das Hauptprogramm oder den aktuellen Handlungsstrang zurückgeben soll.


def nextSquare(x, limit):
    while x <= limit:
        yield x*x
        x += 1


sequenz = nextSquare(2, 4)
print(list(sequenz))
# [4, 9, 16]
sequenz = nextSquare(4, 6)
print(next(sequenz))
# 16
print(next(sequenz))
# 25

Im obigen Beispiel ist ersichtlich, dass die Umwandlung des iterierbaren Objekts in eine Liste, gleich alle durch den Iterator ansprechbare Werte zurückgibt. Soll auf den nächsten Wert zurückgegriffen werden, so wird next benutzt.

Vom Aufbau her wird einfach eine Liste von yield Anweisungen abgearbeitet:


def gen():
    yield 'A'
    yield 'B'
    yield 'C'


for i in gen():
    print(i)
# A
# B
# C

An einen Generator können Werte auch gesendet werden. Das ist dann nützlich, wenn eine Sequenz ab einem gewissen Wert unterbrochen werden und dann für einen höheren Wert wieder fortgesetzt werden soll:


def double_number(number):
    while True:
        number *= 2
        number = yield number


c = double_number(4)
print(next(c))
# 8
# print(next(c))
# Dies wuerde zu einer Exception fuehren

print(c.send(5))
# 10
print(c.send(1500))
# 3000
print(c.send(3))
# 6

Damit wäre dieser Blogartikel schon beendet. Viel Spass beim Üben!

Kommentare

Beliebte Posts aus diesem Blog

Azure Service Principal

  Azure Service principal Ein Azure Service Principal ist ist eine im Azure Portal definierte Identität, die von Automatisierungstools und Services gebraucht wird, um auf spezifische Ressourcen innerhalb der Azure Cloud zuzugreifen. Die Identität verfügt über ein Login mit Passwort oder aber über ein Zertifikat, um sich gegenüber Azure zu authentifizieren. Da die Identität über eine spezfische Rolle verfügt kann die Autorisierung gegenüber ausführbarer Funktionen fein granular gesteuert werden. Mittels der Identität kann dann beispielsweise eine in Azure DevOps entwickelte App, automatisch auf ein Server in der Azure Cloud deployt werden. Service connection In Azure DevOps muss eine Service Connection erstellt werden, so dass Azure DevOps mit den Angaben des Service Prinicpals automatisch eine Anwendung deployen kann. Das Vorgehen für die Erstellung eines Service Principals ist wie folgt: Projects auswählen Zuerst wird in den "Organization Settings" der...

Python : Einführung in pygame

Einführung Pygame ist eine Bibliothek in Python, die es erlaubt grafisch anspruchsvolle Spiele zu erstellen. Damit pygame effektiv eingesetzt werden kann, braucht es doch einiges an Basiswissen, das in diesem Blog vermittelt wird. Gerade für Spiele wird vielmals von Elementen der Künstlichen Intelligenz Gebrauch gemacht. Teilweise handelt es sich nur um offensichtliche KI-Routinen aber andererseits auch um hochkomplexe Algorithmen, die nicht so trivial zu verstehen sind. Vor etlichen Jahren habe ich ein Buch gekauft, mit dem ansprechenden Namen " AI for Game developers " von David M. Bourg und Glenn Seemann. Das Ziel war es, herauszufinden, wie einfach ein Spiel unter Java 5 realisiert werden könnte. Das ganze sollte ein Kundenprojekt werden, ist aber dann nie zustande gekommen, weil der Aufwand zu gross war, das Projekt mit den bestehenden Java Bibliotheken umzusetzen. Für meine weitere "Forschung" hinsichtlich KI habe ich dieses Buch wieder zur Hand genomm...

Python: Einführung in pygame - Intelligenter Gegenspieler

Einführung In meinem ersten Blog zum Thema "Pygame" - Python: Eine Einführung in pygame - habe ich kurz erklärt wie Animationen mittels pygame realisiert werden können. Auch ein fundamentales Thema, nämlich die Kollisionserkennung wurde kurz behandelt. In diesem zweiten Blog, befassen wir uns intensiv mit dem Thema "Chasing and Evading", was übersetzt heisst: Jagen und Ausweichen. Dabei geht es um zwei Objekte, die sich animiert über das Surface (Windows-Oberfläche des Spiels) bewegen. Die zwei Objekte heissen im Spielekontext Prey (Opfer/Beute) und Predator (Raubtier/Plünderer). Dabei ist der computer kontrollierte Spieler meistens der Predator und der Prey, der vom Gamer kontrollierte Spieler.  Chasing and Evading wird schon in den Kontext von Künstlicher Intelligenz gestellt, obschon im Hintergrund kein grosses "neuronales Netz" oder irgendein selbst lernender Algorithmus steckt. Die Verfolgung durch den Computer basiert eigentlich auf einem C...