Heim >Backend-Entwicklung >Python-Tutorial >Python-Multithreading

Python-Multithreading

高洛峰
高洛峰Original
2016-11-23 13:59:571191Durchsuche

Multi-Threading ähnelt der Ausführung mehrerer verschiedener Programme gleichzeitig und bietet folgende Vorteile:

Durch die Verwendung von Threads können Aufgaben in Langzeitprogrammen zur Verarbeitung in den Hintergrund gerückt werden.

Die Benutzeroberfläche kann attraktiver gestaltet werden, sodass, wenn der Benutzer auf eine Schaltfläche klickt, um die Verarbeitung bestimmter Ereignisse auszulösen, ein Fortschrittsbalken angezeigt werden kann, der den Fortschritt der Verarbeitung anzeigt.

Die Die Laufgeschwindigkeit des Programms kann beschleunigt werden

Threads sind nützlicher bei der Implementierung einiger Warteaufgaben wie Benutzereingaben, Lesen und Schreiben von Dateien sowie Senden und Empfangen von Daten über das Netzwerk. In diesem Fall können wir einige wertvolle Ressourcen wie Speichernutzung usw. freigeben.

Threads unterscheiden sich immer noch von Prozessen während der Ausführung. Jeder unabhängige Thread verfügt über einen Einstiegspunkt für die Programmausführung, eine sequentielle Ausführungssequenz und einen Ausstiegspunkt für das Programm. Threads können jedoch nicht unabhängig ausgeführt werden und müssen im Anwendungsprogramm vorhanden sein, und das Anwendungsprogramm bietet eine Steuerung für die Ausführung mehrerer Threads.

Jeder Thread verfügt über einen eigenen Satz von CPU-Registern, den sogenannten Thread-Kontext, der den Zustand der CPU-Register zum letzten Mal widerspiegelt, als der Thread ausgeführt wurde.

Der Befehlszeiger und das Stapelzeigerregister sind die beiden wichtigsten Register im Thread-Kontext. Der Thread wird immer im Prozesskontext ausgeführt. Diese Adressen werden verwendet, um den Speicher im Adressraum des Prozesses zu markieren der Thread.

Threads können vorbelegt (unterbrochen) werden.

Ein Thread kann vorübergehend angehalten werden (auch Ruhezustand genannt), während andere Threads ausgeführt werden – dies wird als Thread-Zurückziehen bezeichnet.

Beginnen Sie mit dem Erlernen von Python-Threads

Es gibt zwei Möglichkeiten, Threads in Python zu verwenden: Funktionen oder Klassen zum Umschließen von Thread-Objekten.

Funktional: Rufen Sie die Funktion start_new_thread() im Thread-Modul auf, um einen neuen Thread zu generieren. Die Syntax lautet wie folgt:

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

Parameterbeschreibung:

Funktion – Thread-Funktion.

args – die an die Thread-Funktion übergebenen Parameter, es muss ein Tupeltyp sein.

kwargs – optionale Parameter.

Instanz:

#!/usr/bin/python

Thread importieren

Importzeit

# Definieren Sie eine Funktion für den Thread

def print_time(threadName, Verzögerung):

count = 0

while count < 5:

time.sleep(delay)

count = 1

print "%s: %s" % (threadName, time . ctime(time.time()) )

# Erstelle zwei Threads

versuche:

thread.start_new_thread( print_time, ("Thread- 1", 2, ) )

thread.start_new_thread( print_time, ("Thread-2", 4, ) )

außer:

print "Fehler: nicht möglich start thread"

while 1:

pass

Das Ausgabeergebnis der oben genannten Ausführung Das Programm lautet wie folgt:

Thread-1: Do, 22. Jan. 15:42:17 2009

Thread-1: Do, 22. Jan. 15 :42:19 2009

Thread-2: Do, 22. Jan. 15:42:19 2009

Thread-1: Do, 22. Jan. 15:42:21 2009

Thread -2: Do, 22. Jan. 15:42:23 2009

Thread-1: Do, 22. Jan. 15:42:23 2009

Thread-1: Do, 22. Jan. 15:42:25 2009

Thread-2: Do, 22. Jan. 15:42:27 2009

Thread-2: Do, 22. Jan. 15:42:31 2009

Thread-2: Do. Jan 22 15:42:35 2009

Das Ende eines Threads hängt im Allgemeinen vom natürlichen Ende der Thread-Funktion ab; Sie können auch thread.exit() aufrufen; in der Thread-Funktion, die eine SystemExit-Ausnahme auslöst, um den Zweck des Beendens des Threads zu erreichen.

Threading-Modul

Python bietet Unterstützung für Threads durch zwei Standardbibliotheken, Thread und Threading. Thread bietet einfache Low-Level-Threads und eine einfache Sperre.

Andere vom Thread-Modul bereitgestellte Methoden:

threading.currentThread(): Gibt die aktuelle Thread-Variable zurück.

threading.enumerate(): Gibt eine Liste mit laufenden Threads zurück. „Laufen“ bezieht sich auf die Zeit nach dem Start und vor dem Ende des Threads, mit Ausnahme von Threads vor dem Start und nach der Beendigung.

threading.activeCount(): Gibt die Anzahl der laufenden Threads zurück, was das gleiche Ergebnis wie len(threading.enumerate()) hat.

Zusätzlich zu den Verwendungsmethoden stellt das Thread-Modul auch die Thread-Klasse zur Verarbeitung von Threads bereit. Die Thread-Klasse stellt die folgenden Methoden bereit:

run(): Eine Methode zur Darstellung der Thread-Aktivität.

start(): Thread-Aktivität starten.

join([time]): Warten Sie, bis der Thread beendet ist. Dadurch wird der aufrufende Thread blockiert, bis die Methode „join()“ des Threads abgebrochen wird – normal beendet wird oder eine nicht behandelte Ausnahme ausgelöst wird – oder ein optionaler Timeout auftritt.

isAlive(): Gibt zurück, ob der Thread aktiv ist.

getName(): Gibt den Thread-Namen zurück.

setName(): Setzt den Thread-Namen.

Verwenden Sie das Threading-Modul, um einen Thread zu erstellen

Verwenden Sie das Threading-Modul, um einen Thread zu erstellen, erben Sie direkt von threading.Thread und überschreiben Sie dann die __init__-Methode und die Ausführungsmethode:

#!/usr/bin/python

Threading importieren

Importzeit

exitFlag = 0

class myThread (threading.Thread): #Inherit the parent class threading.Thread

def __init__(self, threadID, Name, Zähler):

                                                                                                                                     not not been been there —                                                 . Der Thread führt die Ausführungsfunktion direkt nach seiner Erstellung aus

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:

thread.exit()

time.sleep(delay)

print „%s: %s“ % (threadName, time.ctime(time.time()))

counter -= 1

# Einen neuen Thread erstellen

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

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

# Thread öffnen

thread1.start()

thread2. start()

print „Hauptthread verlassen“

Die Ausführungsergebnisse des obigen Programms sind wie folgt;

Thread-1: Do, 21. März 09:10:03 2013

Thread-1: Do, 21. März 09:10:04 2013

Thread-2: Do. März 21 09:10:04 2013

Thread-1: Do 21. März 09:10:05 2013

Thread-1: Do 21. März 09:10:06 2013

Thread-2: Do, 21. März 09:10:06 2013

Thread-1: Do, 21. März 09:10:07 2013

Thread-1 wird verlassen

Thread -2: Do, 21. März 09:10:08 2013

Thread-2: Do, 21. März 09:10:10 2013

Thread-2: Do, 21. März 09:10:12 2013

Thread-2 beenden

Thread-Synchronisation

Wenn mehrere Threads gemeinsam bestimmte Daten ändern, kann es zu unvorhersehbaren Ergebnissen kommen Um die Richtigkeit der Daten sicherzustellen, müssen mehrere Threads synchronisiert werden.

Eine einfache Thread-Synchronisierung kann mithilfe der Sperr- und Rlock-Methoden des Thread-Objekts erreicht werden. Für Daten, für deren Ausführung jeweils nur ein Thread erforderlich ist, können die Vorgänge zwischen Erwerb und Freigabe platziert werden Methoden. Wie folgt:

Der Vorteil von Multithreading besteht darin, dass mehrere Aufgaben gleichzeitig ausgeführt werden können (zumindest fühlt es sich so an). Wenn Threads jedoch Daten gemeinsam nutzen müssen, kann es zu Problemen mit der Datensynchronisierung kommen.

Stellen Sie sich diese Situation vor: Alle Elemente in einer Liste sind 0, der Thread „set“ ändert alle Elemente von hinten nach vorne auf 1 und der Thread „print“ ist dafür verantwortlich, die Liste von vorne nach hinten zu lesen und Drucken.

Wenn sich dann der Thread „set“ zu ändern beginnt, druckt der Thread „print“ möglicherweise die Liste und die Ausgabe ist halb 0 und halb 1. Dies ist die Desynchronisation der Daten. Um diese Situation zu vermeiden, wurde das Konzept der Sperren eingeführt.

Schlösser haben zwei Zustände – gesperrt und entsperrt. Immer wenn ein Thread wie „set“ auf gemeinsam genutzte Daten zugreifen möchte, muss er zuerst die Sperre erhalten. Wenn ein anderer Thread wie „print“ bereits eine Sperre erhalten hat, muss der Thread „set“ angehalten werden, was eine synchrone Blockierung darstellt bis der Thread " Drucken "Nachdem der Zugriff abgeschlossen und die Sperre aufgehoben wurde, lassen Sie den Thread weiter "setzen".

Nach einer solchen Verarbeitung werden beim Drucken der Liste entweder alle Nullen oder alle Einsen ausgegeben, und es gibt keine peinliche Szene mehr aus halben Nullen und halben Einsen mehr.

Instanz:

#!/usr/bin/python

Threading importieren

Importzeit

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

          # Erhalten Sie die Sperre und geben Sie „True“ zurück nach Zeitüberschreitung zurückgegeben > threadLock.release()

def print_time(threadName, delay, counter):

while counter:

time.sleep(delay)

print "%s: %s" % (threadName, time.ctime(time.time()))

counter -= 1

threadLock = threading.Lock()

threads = []

# Einen neuen Thread erstellen

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

thread2 = myThread( 2, „Thread-2“, 2)

# Einen neuen Thread starten

thread1.start()

thread2.start()

# Thread zur Threadliste hinzufügen

threads.append(thread1)

threads.append (thread2)

# Warten, bis alle Threads abgeschlossen sind

für t in Threads:

t.join()

print „ Exiting Main Thread“

Thread Priority Queue (Queue)

Pythons Queue-Modul bietet eine synchrone, threadsichere Warteschlangenklasse, einschließlich FIFO-Warteschlange (First In First Out), LIFO ( (Last In First Out) Warteschlange LifoQueue und Prioritätswarteschlange PriorityQueue. Diese Warteschlangen implementieren Sperrprimitive und können direkt in Multithreads verwendet werden. Warteschlangen können verwendet werden, um eine Synchronisierung zwischen Threads zu erreichen.

Gemeinsame Methoden im Queue-Modul:

Queue.qsize() gibt die Größe der Warteschlange zurück

Queue.empty() Wenn die Warteschlange ist leer, gibt True zurück, andernfalls False

Queue.full() Wenn die Warteschlange voll ist, wird True zurückgegeben, andernfalls False

Queue.full entspricht der maximalen Größe

Queue.get([ block[, timeout]]) Holen Sie sich die Warteschlange, Timeout-Wartezeit

Queue.get_nowait() entspricht Queue.get(False)

Queue.put(item ) In die Warteschlange schreiben, Timeout-Wartezeit

Queue.put_nowait(item) entspricht Queue.put(item, False)

Queue.task_done() Nach Abschluss einer Aufgabe wird die Warteschlange Die Funktion .task_done() gibt eine Nachricht an die abgeschlossene Aufgabe zurück

Queue.join() bedeutet eigentlich, zu warten, bis die Warteschlange leer ist, bevor andere Vorgänge ausgeführt werden

Beispiel:

#!/usr/bin/python

Import Queue

Import Threading

Importzeit

exitFlag = 0

class myThread (threading.Thread):

def __init__(self, threadID, name, q):

                                                                                                                             

process_data(self.name, self.q)

print „Exiting“ self.name

def Process_data(threadName, q):

while not exitFlag:

queueLock.acquire()

if not workQueue.empty( ):

data = q.get()

queueLock.release()

            print "%s processing %s" % (threadName, data)

        else:

            queueLock.release()

        time.sleep(1)

 

threadList = ["Thread-1", "Thread-2", "Thread-3"]

nameList = ["One", "Two", "Three ", "Vier", "Fünf"]

queueLock = threading.Lock()

workQueue = Queue.Queue(10)

threads = []

threadID = 1

 

# 创建新线程

für tName in threadList:

    thread = myThread(threadID, tName, workQueue)

    thread.start()

    threads.append(thread)

    threadID  = 1

 

# 填充队列

queueLock.acquire()

für Wort in

# 等待队列清空

while nicht 🎜>

exitFlag = 1

 

# 等待所有线程完成

für t in-Threads:

    t.join()

print "Hauptthread verlassen"

   

以上程序执行结果:

Startthread-1

Thread-2 wird gestartet

Thread-3 wird gestartet

Thread-1 verarbeitet Eins

Thread-2 verarbeitet Zwei

Thread-3 verarbeitet Drei

Thread-1 verarbeitet Vier

Thread-2 verarbeitet Fünf

Thread-3 wird beendet

Thread-1 wird beendet

Thread-2 wird beendet

Hauptthread verlassen

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Vorheriger Artikel:Python-XML-AnalyseNächster Artikel:Python-XML-Analyse