Implementasi Threads

Di bab ini, kita akan belajar bagaimana mengimplementasikan utas dengan Python.

Modul Python untuk Implementasi Thread

Utas Python terkadang disebut proses ringan karena utas menempati lebih sedikit memori daripada proses. Utas memungkinkan melakukan banyak tugas sekaligus. Dengan Python, kami memiliki dua modul berikut yang mengimplementasikan utas dalam sebuah program -

  • <_thread>module

  • <threading>module

Perbedaan utama antara kedua modul ini adalah <_thread> modul memperlakukan utas sebagai fungsi sedangkan, <threading>modul memperlakukan setiap utas sebagai objek dan mengimplementasikannya dengan cara berorientasi objek. Apalagi<_thread>modul efektif dalam penguliran tingkat rendah dan memiliki kemampuan yang lebih sedikit daripada modul <threading> modul.

<_thread> modul

Di versi Python sebelumnya, kami memiliki file <thread>modul tetapi telah dianggap sebagai "usang" untuk waktu yang cukup lama. Pengguna didorong untuk menggunakan<threading>modul sebagai gantinya. Oleh karena itu, dalam Python 3 modul "thread" tidak tersedia lagi. Ini telah diganti namanya menjadi "<_thread>"untuk ketidakcocokan mundur dengan Python3.

Untuk menghasilkan utas baru dengan bantuan <_thread> modul, kita perlu memanggil start_new_threadmetode itu. Cara kerja metode ini dapat dipahami dengan bantuan sintaks berikut -

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

Di sini -

  • args adalah tuple argumen

  • kwargs adalah kamus opsional untuk argumen kata kunci

Jika kita ingin memanggil fungsi tanpa memberikan argumen maka kita perlu menggunakan tuple argumen kosong di args.

Panggilan metode ini segera kembali, utas anak dimulai, dan memanggil fungsi dengan daftar yang diteruskan, jika ada, dari args. Utas berakhir saat dan saat fungsi kembali.

Contoh

Berikut adalah contoh untuk menghasilkan utas baru dengan menggunakan <_thread>modul. Kami menggunakan metode start_new_thread () di sini.

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

Keluaran

Keluaran berikut akan membantu kita memahami pembuatan utas baru dengan bantuan <_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

Modul <threading>

Itu <threading>modul mengimplementasikan dengan cara berorientasi objek dan memperlakukan setiap utas sebagai objek. Oleh karena itu, modul ini memberikan dukungan tingkat tinggi yang jauh lebih bertenaga untuk utas daripada modul <_thread>. Modul ini disertakan dengan Python 2.4.

Metode tambahan dalam modul <threading>

Itu <threading> modul terdiri dari semua metode <_thread>modul tetapi menyediakan metode tambahan juga. Metode tambahannya adalah sebagai berikut -

  • threading.activeCount() - Metode ini mengembalikan jumlah objek utas yang aktif

  • threading.currentThread() - Metode ini mengembalikan jumlah objek utas dalam kontrol utas pemanggil.

  • threading.enumerate() - Metode ini mengembalikan daftar semua objek utas yang saat ini aktif.

  • Untuk mengimplementasikan threading, file <threading> modul memiliki Thread kelas yang menyediakan metode berikut -

    • run() - Metode run () adalah titik masuk untuk utas.

    • start() - Metode start () memulai utas dengan memanggil metode run.

    • join([time]) - Join () menunggu utas berakhir.

    • isAlive() - Metode isAlive () memeriksa apakah utas masih dijalankan.

    • getName() - Metode getName () mengembalikan nama utas.

    • setName() - Metode setName () menetapkan nama utas.

Bagaimana cara membuat utas menggunakan modul <threading>?

Di bagian ini, kita akan belajar cara membuat utas menggunakan <threading>modul. Ikuti langkah-langkah ini untuk membuat utas baru menggunakan modul <threading> -

  • Step 1 - Pada langkah ini, kita perlu mendefinisikan subclass baru dari Thread kelas.

  • Step 2 - Kemudian untuk menambahkan argumen tambahan, kita perlu mengganti __init__(self [,args]) metode.

  • Step 3 - Dalam langkah ini, kita perlu mengganti metode run (self [, args]) untuk mengimplementasikan apa yang harus dilakukan thread saat dimulai.

  • Sekarang, setelah membuat file Thread subclass, kita dapat membuat turunannya dan kemudian memulai utas baru dengan memanggil start(), yang pada gilirannya memanggil run() metode.

Contoh

Pertimbangkan contoh ini untuk mempelajari cara membuat utas baru dengan menggunakan <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

Keluaran

Sekarang, perhatikan output berikut -

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

Program Python untuk Berbagai Status Thread

Ada lima status utas - baru, dapat dijalankan, berjalan, menunggu, dan mati. Di antara lima Dari lima ini, kami akan berfokus pada tiga kondisi - berlari, menunggu dan mati. Sebuah utas mendapatkan sumber dayanya dalam keadaan berjalan, menunggu sumber daya dalam keadaan menunggu; rilis final sumber daya, jika menjalankan dan diperoleh dalam keadaan mati.

Program Python berikut dengan bantuan metode start (), sleep () dan join () akan menunjukkan bagaimana sebuah thread dimasukkan masing-masing dalam keadaan berjalan, menunggu dan mati.

Step 1 - Impor modul yang diperlukan, <threading> dan <time>

import threading
import time

Step 2 - Tentukan fungsi, yang akan dipanggil saat membuat utas.

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

Step 3 - Kami menggunakan metode sleep () modul waktu untuk membuat utas kami menunggu, katakanlah 2 detik.

time.sleep(2)

Step 4 - Sekarang, kami membuat utas bernama T1, yang mengambil argumen dari fungsi yang ditentukan di atas.

T1 = threading.Thread(target=thread_states)

Step 5- Sekarang, dengan bantuan fungsi start () kita bisa memulai utas kita. Ini akan menghasilkan pesan, yang telah kita atur saat mendefinisikan fungsinya.

T1.start()
Thread entered in running state

Step 6 - Sekarang, akhirnya kita bisa mematikan utas dengan metode join () setelah itu menyelesaikan eksekusinya.

T1.join()

Memulai utas dengan Python

Dalam python, kita dapat memulai utas baru dengan berbagai cara tetapi yang termudah di antara mereka adalah dengan mendefinisikannya sebagai satu fungsi. Setelah mendefinisikan fungsinya, kita bisa meneruskan ini sebagai target untuk yang baruthreading.Threadobjek dan sebagainya. Jalankan kode Python berikut untuk memahami cara kerja fungsinya -

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())

Keluaran

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

Utas daemon dengan Python

Sebelum mengimplementasikan utas daemon dengan Python, kita perlu tahu tentang utas daemon dan penggunaannya. Dalam hal komputasi, daemon adalah proses latar belakang yang menangani permintaan untuk berbagai layanan seperti pengiriman data, transfer file, dll. Akan tidak aktif jika tidak diperlukan lagi. Tugas yang sama dapat dilakukan dengan bantuan utas non-daemon juga. Namun, dalam kasus ini, utas utama harus melacak utas non-daemon secara manual. Di sisi lain, jika kita menggunakan utas daemon maka utas utama dapat sepenuhnya melupakan hal ini dan akan dimatikan ketika utas utama keluar. Poin penting lainnya tentang utas daemon adalah kita dapat memilih untuk menggunakannya hanya untuk tugas-tugas non-esensial yang tidak akan memengaruhi kita jika tidak selesai atau terhenti di antaranya. Berikut adalah implementasi utas daemon di 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()

Pada kode diatas terdapat dua fungsi yaitu >nondaemonThread() dan >daemonThread(). Fungsi pertama mencetak statusnya dan tidur setelah 8 detik sementara fungsi deamonThread () mencetak Halo setelah setiap 2 detik tanpa batas. Kita dapat memahami perbedaan antara utas nondaemon dan daemon dengan bantuan keluaran berikut -

Hello

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