サイトアイコン 【TechGrowUp】

Python開発入門56 Pythonのthreadingモジュール入門!スレッドで効率的な並行処理を学ぶ

はじめに

Pythonのthreadingモジュールは、軽量な並行処理を実現するための標準ライブラリです。スレッドを使用することで、複数のタスクを同時に実行し、プログラムのパフォーマンスを向上させることが可能です。

この記事では、スレッドの基本的な使い方から、より高度な制御方法までを初心者向けに解説します。

threadingモジュールとは

threadingモジュールの概要

threadingモジュールは、Pythonでスレッドベースの並行処理を実現するためのモジュールです。スレッドは、同一プロセス内で動作する軽量な処理単位で、以下のような特徴を持っています:

特徴:

threadingモジュールの基本的な使い方

基本的なスレッドの作成

スレッドを作成するには、threading.Threadクラスを使用します。

例:スレッドの基本的な使用方法
import threading

def print_numbers():
    for i in range(5):
        print(f"番号: {i}")

# スレッドを作成
thread = threading.Thread(target=print_numbers)

# スレッドを開始
thread.start()

# スレッドの終了を待機
thread.join()

説明:

スレッドクラスの継承

Threadクラスを継承して独自のスレッドを作成することも可能です。

例:Threadクラスを継承する方法
import threading

class MyThread(threading.Thread):
    def run(self):
        for i in range(5):
            print(f"スレッド: {i}")

# スレッドインスタンスを作成
thread = MyThread()

# スレッドを開始
thread.start()

# スレッドの終了を待機
thread.join()

スレッドの制御

スレッドの状態確認

スレッドの状態を確認するには、is_aliveメソッドを使用します。

例:スレッドの状態を確認
thread = threading.Thread(target=print_numbers)
thread.start()
print(f"スレッドの状態: {thread.is_alive()}")
thread.join()
print(f"スレッドの状態: {thread.is_alive()}")

デーモンスレッド

デーモンスレッドは、メインスレッドが終了すると強制的に終了します。

例:デーモンスレッドの使用
def print_numbers():
    for i in range(10):
        print(i)

thread = threading.Thread(target=print_numbers, daemon=True)
thread.start()

スレッド間の同期

ロック(Lock)

スレッド間でデータ競合を防ぐためにロックを使用します。

例:ロックの使用
import threading

lock = threading.Lock()
counter = 0

def increment():
    global counter
    with lock:
        local_counter = counter
        local_counter += 1
        counter = local_counter

threads = [threading.Thread(target=increment) for _ in range(10)]

for thread in threads:
    thread.start()

for thread in threads:
    thread.join()

print(f"カウンターの値: {counter}")

条件変数(Condition)

特定の条件でスレッドを待機させる場合に使用します。

例:条件変数の使用
import threading

condition = threading.Condition()

def worker():
    with condition:
        print("スレッドが待機中...")
        condition.wait()
        print("スレッドが再開しました!")

def notifier():
    with condition:
        print("通知を送信します...")
        condition.notify_all()

worker_thread = threading.Thread(target=worker)
notifier_thread = threading.Thread(target=notifier)

worker_thread.start()
notifier_thread.start()

worker_thread.join()
notifier_thread.join()

threadingを使った応用例

タスクの並行実行

複数のタスクを並行して実行します。

import threading

def task(name, delay):
    for i in range(3):
        print(f"{name}: 実行中 {i}")
        time.sleep(delay)

threads = [
    threading.Thread(target=task, args=("タスク1", 1)),
    threading.Thread(target=task, args=("タスク2", 2)),
]

for thread in threads:
    thread.start()

for thread in threads:
    thread.join()

プログレスバーの更新

スレッドを使ってプログレスバーを更新する処理を実装します。

import threading
import time

def update_progress():
    for i in range(10):
        print(f"進行状況: {i+1}/10")
        time.sleep(1)

thread = threading.Thread(target=update_progress)
thread.start()
thread.join()

threadingの注意点

  1. GILの影響
    PythonのスレッドはGILの制約を受けるため、CPUバウンドタスクには適していません。
  2. データ競合
    複数のスレッドが同じデータにアクセスする場合は、ロックや同期を適切に使用する必要があります。
  3. デバッグの難しさ
    並行処理におけるエラーは再現が難しいため、デバッグに時間がかかる場合があります。

まとめ

Pythonのthreadingモジュールは、軽量なスレッド処理を実現するための便利なツールです。この記事で解説した基本的な使い方や同期方法を参考に、スレッドを活用して効率的なプログラムを作成してください!

モバイルバージョンを終了