Vai al contenuto

Inferenza thread-safe con i modelli di YOLO

L'esecuzione dei modelli YOLO in un ambiente multi-thread richiede un'attenta considerazione per garantire la sicurezza dei thread. Python's threading permette di eseguire più thread in contemporanea, ma quando si tratta di usare i modelli di YOLO tra questi thread, ci sono importanti problemi di sicurezza da tenere presenti. Questa pagina vi guiderà nella creazione di inferenze di modelli YOLO sicure per i thread.

Comprendere la filettatura di Python

Python I thread sono una forma di parallelismo che consente al programma di eseguire più operazioni contemporaneamente. Tuttavia, il blocco globale dell'interprete (GIL) di Python significa che solo un thread può eseguire il bytecode di Python alla volta.

Esempi di thread singoli o multipli

Sebbene questo sembri un limite, i thread possono comunque garantire la concomitanza, soprattutto per le operazioni legate all'I/O o quando si usano operazioni che rilasciano il GIL, come quelle eseguite dalle librerie C sottostanti a YOLO.

Il pericolo delle istanze di modello condivise

L'istanziazione di un modello YOLO al di fuori dei propri thread e la condivisione di questa istanza tra più thread può portare a condizioni di gara, in cui lo stato interno del modello viene modificato in modo incoerente a causa di accessi simultanei. Questo è particolarmente problematico quando il modello o i suoi componenti contengono uno stato che non è stato progettato per essere thread-safe.

Esempio non thread-safe: Istanza di un singolo modello

Quando si usano i thread in Python, è importante riconoscere gli schemi che possono portare a problemi di concorrenza. Ecco cosa si dovrebbe evitare: condividere una singola istanza del modello YOLO tra più thread.

# Unsafe: Sharing a single model instance across threads
from threading import Thread

from ultralytics import YOLO

# Instantiate the model outside the thread
shared_model = YOLO("yolo11n.pt")


def predict(image_path):
    """Predicts objects in an image using a preloaded YOLO model, take path string to image as argument."""
    results = shared_model.predict(image_path)
    # Process results


# Starting threads that share the same model instance
Thread(target=predict, args=("image1.jpg",)).start()
Thread(target=predict, args=("image2.jpg",)).start()

Nell'esempio precedente, l'elemento shared_model è utilizzato da più thread, il che può portare a risultati imprevedibili perché predict potrebbe essere eseguito simultaneamente da più thread.

Esempio non thread-safe: Istanze multiple del modello

Allo stesso modo, ecco un modello non sicuro con istanze multiple del modello YOLO :

# Unsafe: Sharing multiple model instances across threads can still lead to issues
from threading import Thread

from ultralytics import YOLO

# Instantiate multiple models outside the thread
shared_model_1 = YOLO("yolo11n_1.pt")
shared_model_2 = YOLO("yolo11n_2.pt")


def predict(model, image_path):
    """Runs prediction on an image using a specified YOLO model, returning the results."""
    results = model.predict(image_path)
    # Process results


# Starting threads with individual model instances
Thread(target=predict, args=(shared_model_1, "image1.jpg")).start()
Thread(target=predict, args=(shared_model_2, "image2.jpg")).start()

Anche se ci sono due istanze di modello separate, il rischio di problemi di concorrenza esiste ancora. Se l'implementazione interna di YOLO non è thread-safe, l'uso di istanze separate potrebbe non prevenire le condizioni di gara, soprattutto se queste istanze condividono risorse o stati sottostanti che non sono thread-local.

Inferenza sicura per i thread

Per eseguire un'inferenza thread-safe, è necessario istanziare un modello YOLO separato all'interno di ogni thread. Questo assicura che ogni thread abbia la propria istanza isolata del modello, eliminando il rischio di condizioni di gara.

Esempio sicuro per i thread

Ecco come istanziare un modello YOLO all'interno di ciascun thread per un'inferenza parallela sicura:

# Safe: Instantiating a single model inside each thread
from threading import Thread

from ultralytics import YOLO


def thread_safe_predict(image_path):
    """Predict on an image using a new YOLO model instance in a thread-safe manner; takes image path as input."""
    local_model = YOLO("yolo11n.pt")
    results = local_model.predict(image_path)
    # Process results


# Starting threads that each have their own model instance
Thread(target=thread_safe_predict, args=("image1.jpg",)).start()
Thread(target=thread_safe_predict, args=("image2.jpg",)).start()

In questo esempio, ogni thread crea il proprio YOLO istanza. Questo impedisce a qualsiasi thread di interferire con lo stato del modello di un altro, garantendo così che ogni thread esegua l'inferenza in modo sicuro e senza interazioni impreviste con gli altri thread.

Conclusione

Quando si utilizzano i modelli YOLO con Python's threadingPer garantire la sicurezza dei thread, i modelli devono essere sempre istanziati all'interno del thread che li utilizzerà. Questa pratica evita le condizioni di gara e assicura che le operazioni di inferenza vengano eseguite in modo affidabile.

Per scenari più avanzati e per ottimizzare ulteriormente le prestazioni dell'inferenza multi-thread, si può prendere in considerazione l'uso del parallelismo basato sul processo con multiprocessing o sfruttando una coda di attività con processi worker dedicati.

FAQ

Come si possono evitare le condizioni di gara quando si utilizzano i modelli YOLO in un ambiente Python multi-thread?

Per evitare condizioni di gara quando si usano i modelli Ultralytics YOLO in un ambiente Python multi-thread, istanziare un modello YOLO separato in ogni thread. Questo assicura che ogni thread abbia la propria istanza isolata del modello, evitando la modifica concorrente dello stato del modello.

Esempio:

from threading import Thread

from ultralytics import YOLO


def thread_safe_predict(image_path):
    """Predict on an image in a thread-safe manner."""
    local_model = YOLO("yolo11n.pt")
    results = local_model.predict(image_path)
    # Process results


Thread(target=thread_safe_predict, args=("image1.jpg",)).start()
Thread(target=thread_safe_predict, args=("image2.jpg",)).start()

Per ulteriori informazioni su come garantire la sicurezza delle filettature, visitare Thread-Safe Inference with YOLO Models.

Quali sono le migliori pratiche per eseguire l'inferenza del modello YOLO in multi-thread in Python?

Per eseguire in modo sicuro l'inferenza del modello YOLO in multi-thread in Python, seguite queste best practice:

  1. Istanziare i modelli di YOLO all'interno di ciascun thread, anziché condividere una singola istanza di modello tra i thread.
  2. Utilizzare Python's multiprocessing per l'elaborazione parallela, per evitare problemi legati al Global Interpreter Lock (GIL).
  3. Rilascia il GIL utilizzando le operazioni eseguite dalle librerie C sottostanti a YOLO.

Esempio di istanziazione del modello thread-safe:

from threading import Thread

from ultralytics import YOLO


def thread_safe_predict(image_path):
    """Runs inference in a thread-safe manner with a new YOLO model instance."""
    model = YOLO("yolo11n.pt")
    results = model.predict(image_path)
    # Process results


# Initiate multiple threads
Thread(target=thread_safe_predict, args=("image1.jpg",)).start()
Thread(target=thread_safe_predict, args=("image2.jpg",)).start()

Per un ulteriore contesto, consultare la sezione sull'inferenza Thread-Safe.

Perché ogni thread dovrebbe avere la propria istanza del modello YOLO ?

Ogni thread deve avere la propria istanza del modello YOLO per evitare condizioni di gara. Quando una singola istanza del modello è condivisa tra più thread, gli accessi simultanei possono portare a comportamenti imprevedibili e a modifiche dello stato interno del modello. Utilizzando istanze separate, si garantisce l'isolamento dei thread, rendendo le attività multi-thread affidabili e sicure.

Per una guida dettagliata, consultare le sezioni Esempio non thread-safe: Istanza di modello singola e Esempio thread-safe.

In che modo il Global Interpreter Lock (GIL) di Python influisce sull'inferenza del modello di YOLO ?

PythonIl blocco globale dell'interprete (GIL) permette a un solo thread di eseguire il bytecode di Python alla volta, il che può limitare le prestazioni delle operazioni multi-threading legate a CPU. Tuttavia, per le operazioni legate all'I/O o per i processi che utilizzano librerie che rilasciano il GIL, come le librerie C di YOLO, è ancora possibile ottenere la concorrenza. Per migliorare le prestazioni, si può prendere in considerazione l'uso del parallelismo basato sui processi con Python's multiprocessing modulo.

Per ulteriori informazioni sulla filettatura in Python, vedere la sezione Comprendere la filettatura in Python .

È più sicuro usare il parallelismo basato sui processi invece del threading per l'inferenza del modello YOLO ?

Sì, utilizzando Python's multiprocessing è più sicuro e spesso più efficiente per eseguire l'inferenza del modello YOLO in parallelo. Il parallelismo basato sui processi crea spazi di memoria separati, evitando il blocco globale dell'interprete (GIL) e riducendo il rischio di problemi di concorrenza. Ogni processo opererà in modo indipendente con la propria istanza del modello YOLO .

Per ulteriori dettagli sul parallelismo basato sui processi con i modelli di YOLO , consultare la pagina sull'inferenza Thread-Safe.

📅C reato 1 anno fa ✏️ Aggiornato 3 mesi fa

Commenti