Implementierung von Threads

In diesem Kapitel erfahren Sie, wie Sie Threads in Python implementieren.

Python-Modul für die Thread-Implementierung

Python-Threads werden manchmal als einfache Prozesse bezeichnet, da Threads viel weniger Speicher belegen als Prozesse. Mit Threads können mehrere Aufgaben gleichzeitig ausgeführt werden. In Python haben wir die folgenden zwei Module, die Threads in einem Programm implementieren:

  • <_thread>module

  • <threading>module

Der Hauptunterschied zwischen diesen beiden Modulen besteht darin, dass <_thread> Modul behandelt einen Thread als Funktion, während das <threading>Das Modul behandelt jeden Thread als Objekt und implementiert ihn objektorientiert. Darüber hinaus ist die<_thread>Modul ist effektiv in Low-Level-Threading und hat weniger Funktionen als das <threading> Modul.

Modul <_Thread>

In der früheren Version von Python hatten wir die <thread>Modul, aber es wurde für eine ziemlich lange Zeit als "veraltet" angesehen. Benutzer wurden aufgefordert, die zu verwenden<threading>Modul stattdessen. Daher ist in Python 3 das Modul "Thread" nicht mehr verfügbar. Es wurde umbenannt in "<_thread>"für Abwärtsinkompatibilitäten in Python3.

Um einen neuen Thread mit Hilfe des zu generieren <_thread> Modul müssen wir das aufrufen start_new_threadMethode davon. Die Funktionsweise dieser Methode kann mit Hilfe der folgenden Syntax verstanden werden:

_thread.start_new_thread ( function, args[, kwargs] )

Hier -

  • args ist ein Tupel von Argumenten

  • kwargs ist ein optionales Wörterbuch mit Schlüsselwortargumenten

Wenn wir die Funktion aufrufen möchten, ohne ein Argument zu übergeben, müssen wir ein leeres Tupel von Argumenten in verwenden args.

Dieser Methodenaufruf wird sofort zurückgegeben, der untergeordnete Thread wird gestartet und die Funktion mit der übergebenen Liste von Argumenten (falls vorhanden) aufgerufen. Der Thread wird beendet, sobald die Funktion zurückkehrt.

Beispiel

Im Folgenden finden Sie ein Beispiel zum Generieren eines neuen Threads mithilfe von <_thread>Modul. Wir verwenden hier die Methode start_new_thread ().

import _thread
import time

def print_time( threadName, delay):
   count = 0
   while count < 5:
      time.sleep(delay)
      count += 1
      print ("%s: %s" % ( threadName, time.ctime(time.time()) ))

try:
   _thread.start_new_thread( print_time, ("Thread-1", 2, ) )
   _thread.start_new_thread( print_time, ("Thread-2", 4, ) )
except:
   print ("Error: unable to start thread")
while 1:
   pass

Ausgabe

Die folgende Ausgabe hilft uns, die Erzeugung neuer Threads mithilfe von zu verstehen <_thread> Modul.

Thread-1: Mon Apr 23 10:03:33 2018
Thread-2: Mon Apr 23 10:03:35 2018
Thread-1: Mon Apr 23 10:03:35 2018
Thread-1: Mon Apr 23 10:03:37 2018
Thread-2: Mon Apr 23 10:03:39 2018
Thread-1: Mon Apr 23 10:03:39 2018
Thread-1: Mon Apr 23 10:03:41 2018
Thread-2: Mon Apr 23 10:03:43 2018
Thread-2: Mon Apr 23 10:03:47 2018
Thread-2: Mon Apr 23 10:03:51 2018

<Threading> -Modul

Das <threading>Das Modul wird objektorientiert implementiert und behandelt jeden Thread als Objekt. Daher bietet es eine viel leistungsfähigere Unterstützung für Threads auf hoher Ebene als das Modul <_thread>. Dieses Modul ist in Python 2.4 enthalten.

Zusätzliche Methoden im Modul <threading>

Das <threading> Modul umfasst alle Methoden der <_thread>Modul, bietet aber auch zusätzliche Methoden. Die zusätzlichen Methoden sind wie folgt:

  • threading.activeCount() - Diese Methode gibt die Anzahl der aktiven Thread-Objekte zurück

  • threading.currentThread() - Diese Methode gibt die Anzahl der Thread-Objekte im Thread-Steuerelement des Aufrufers zurück.

  • threading.enumerate() - Diese Methode gibt eine Liste aller aktuell aktiven Thread-Objekte zurück.

  • Für die Implementierung von Threading wird die <threading> Modul hat die Thread Klasse, die die folgenden Methoden bereitstellt -

    • run() - Die run () -Methode ist der Einstiegspunkt für einen Thread.

    • start() - Die Methode start () startet einen Thread durch Aufrufen der Methode run.

    • join([time]) - Join () wartet auf das Beenden von Threads.

    • isAlive() - Die Methode isAlive () prüft, ob noch ein Thread ausgeführt wird.

    • getName() - Die Methode getName () gibt den Namen eines Threads zurück.

    • setName() - Die Methode setName () legt den Namen eines Threads fest.

Wie erstelle ich Threads mit dem Modul <threading>?

In diesem Abschnitt erfahren Sie, wie Sie mithilfe von Threads erstellen <threading>Modul. Führen Sie die folgenden Schritte aus, um mit dem Modul <threading> einen neuen Thread zu erstellen.

  • Step 1 - In diesem Schritt müssen wir eine neue Unterklasse der definieren Thread Klasse.

  • Step 2 - Um zusätzliche Argumente hinzuzufügen, müssen wir das überschreiben __init__(self [,args]) Methode.

  • Step 3 - In diesem Schritt müssen wir die Methode run (self [, args]) überschreiben, um zu implementieren, was der Thread beim Starten tun soll.

  • Jetzt nach dem Erstellen der neuen Thread Unterklasse können wir eine Instanz davon erstellen und dann einen neuen Thread starten, indem wir die aufrufen start(), was wiederum die nennt run() Methode.

Beispiel

In diesem Beispiel erfahren Sie, wie Sie mithilfe von einen neuen Thread generieren <threading> Modul.

import threading
import time
exitFlag = 0

class myThread (threading.Thread):
   def __init__(self, threadID, name, counter):
      threading.Thread.__init__(self)
      self.threadID = threadID
      self.name = name
      self.counter = counter
   def run(self):
      print ("Starting " + self.name)
      print_time(self.name, self.counter, 5)
      print ("Exiting " + self.name)
def print_time(threadName, delay, counter):
   while counter:
      if exitFlag:
         threadName.exit()
      time.sleep(delay)
      print ("%s: %s" % (threadName, time.ctime(time.time())))
      counter -= 1

thread1 = myThread(1, "Thread-1", 1)
thread2 = myThread(2, "Thread-2", 2)

thread1.start()
thread2.start()
thread1.join()
thread2.join()
print ("Exiting Main Thread")
Starting Thread-1
Starting Thread-2

Ausgabe

Betrachten Sie nun die folgende Ausgabe:

Thread-1: Mon Apr 23 10:52:09 2018
Thread-1: Mon Apr 23 10:52:10 2018
Thread-2: Mon Apr 23 10:52:10 2018
Thread-1: Mon Apr 23 10:52:11 2018
Thread-1: Mon Apr 23 10:52:12 2018
Thread-2: Mon Apr 23 10:52:12 2018
Thread-1: Mon Apr 23 10:52:13 2018
Exiting Thread-1
Thread-2: Mon Apr 23 10:52:14 2018
Thread-2: Mon Apr 23 10:52:16 2018
Thread-2: Mon Apr 23 10:52:18 2018
Exiting Thread-2
Exiting Main Thread

Python-Programm für verschiedene Thread-Zustände

Es gibt fünf Thread-Zustände - neu, lauffähig, laufend, wartend und tot. Unter diesen fünf werden wir uns hauptsächlich auf drei Staaten konzentrieren - Laufen, Warten und Tot. Ein Thread erhält seine Ressourcen im laufenden Zustand und wartet auf die Ressourcen im wartenden Zustand. Die endgültige Freigabe der Ressource befindet sich, wenn sie ausgeführt und erworben wurde, im toten Zustand.

Das folgende Python-Programm zeigt mit Hilfe der Methoden start (), sleep () und join (), wie ein Thread im laufenden, wartenden und toten Zustand eingegeben wurde.

Step 1 - Importieren Sie die erforderlichen Module <Threading> und <Zeit>

import threading
import time

Step 2 - Definieren Sie eine Funktion, die beim Erstellen eines Threads aufgerufen wird.

def thread_states():
   print("Thread entered in running state")

Step 3 - Wir verwenden das sleep () -Methodenmodul, um unseren Thread etwa 2 Sekunden warten zu lassen.

time.sleep(2)

Step 4 - Jetzt erstellen wir einen Thread mit dem Namen T1, der das Argument der oben definierten Funktion übernimmt.

T1 = threading.Thread(target=thread_states)

Step 5- Jetzt können wir mit Hilfe der Funktion start () unseren Thread starten. Es wird die Nachricht erzeugt, die von uns bei der Definition der Funktion festgelegt wurde.

T1.start()
Thread entered in running state

Step 6 - Jetzt können wir den Thread endlich mit der join () -Methode beenden, nachdem er seine Ausführung abgeschlossen hat.

T1.join()

Starten eines Threads in Python

In Python können wir einen neuen Thread auf verschiedene Arten starten. Am einfachsten ist es jedoch, ihn als einzelne Funktion zu definieren. Nachdem wir die Funktion definiert haben, können wir diese als Ziel für eine neue übergebenthreading.ThreadObjekt und so weiter. Führen Sie den folgenden Python-Code aus, um zu verstehen, wie die Funktion funktioniert:

import threading
import time
import random
def Thread_execution(i):
   print("Execution of Thread {} started\n".format(i))
   sleepTime = random.randint(1,4)
   time.sleep(sleepTime)
   print("Execution of Thread {} finished".format(i))
for i in range(4):
   thread = threading.Thread(target=Thread_execution, args=(i,))
   thread.start()
   print("Active Threads:" , threading.enumerate())

Ausgabe

Execution of Thread 0 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>]

Execution of Thread 1 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>,
      <Thread(Thread-3577, started 3080)>]

Execution of Thread 2 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>,
      <Thread(Thread-3577, started 3080)>,
      <Thread(Thread-3578, started 2268)>]

Execution of Thread 3 started
Active Threads:
   [<_MainThread(MainThread, started 6040)>,
      <HistorySavingThread(IPythonHistorySavingThread, started 5968)>,
      <Thread(Thread-3576, started 3932)>,
      <Thread(Thread-3577, started 3080)>,
      <Thread(Thread-3578, started 2268)>,
      <Thread(Thread-3579, started 4520)>]
Execution of Thread 0 finished
Execution of Thread 1 finished
Execution of Thread 2 finished
Execution of Thread 3 finished

Daemon-Threads in Python

Bevor Sie die Daemon-Threads in Python implementieren, müssen Sie sich mit Daemon-Threads und ihrer Verwendung vertraut machen. In Bezug auf die Datenverarbeitung ist Daemon ein Hintergrundprozess, der die Anforderungen für verschiedene Dienste wie Datenübertragung, Dateiübertragung usw. verarbeitet. Es wäre ruhend, wenn es nicht mehr benötigt wird. Dieselbe Aufgabe kann auch mit Hilfe von Nicht-Daemon-Threads ausgeführt werden. In diesem Fall muss der Hauptthread die Nicht-Daemon-Threads jedoch manuell verfolgen. Wenn wir dagegen Daemon-Threads verwenden, kann der Haupt-Thread dies vollständig vergessen und wird beendet, wenn der Haupt-Thread beendet wird. Ein weiterer wichtiger Punkt bei Daemon-Threads ist, dass wir sie nur für nicht wesentliche Aufgaben verwenden können, die uns nicht betreffen würden, wenn sie nicht abgeschlossen werden oder dazwischen beendet werden. Es folgt die Implementierung von Daemon-Threads in Python -

import threading
import time

def nondaemonThread():
   print("starting my thread")
   time.sleep(8)
   print("ending my thread")
def daemonThread():
   while True:
   print("Hello")
   time.sleep(2)
if __name__ == '__main__':
   nondaemonThread = threading.Thread(target = nondaemonThread)
   daemonThread = threading.Thread(target = daemonThread)
   daemonThread.setDaemon(True)
   daemonThread.start()
   nondaemonThread.start()

Im obigen Code gibt es nämlich zwei Funktionen >nondaemonThread() und >daemonThread(). Die erste Funktion druckt ihren Status und schläft nach 8 Sekunden, während die Funktion deamonThread () alle 2 Sekunden auf unbestimmte Zeit Hallo druckt. Wir können den Unterschied zwischen Nondaemon- und Daemon-Threads anhand der folgenden Ausgabe verstehen:

Hello

starting my thread
Hello
Hello
Hello
Hello
ending my thread
Hello
Hello
Hello
Hello
Hello