C # - Multithreading
UNE threadest défini comme le chemin d'exécution d'un programme. Chaque thread définit un flux de contrôle unique. Si votre application implique des opérations complexes et chronophages, il est souvent utile de définir différents chemins d'exécution ou threads, chaque thread effectuant un travail particulier.
Les fils sont lightweight processes. Un exemple courant d'utilisation de thread est la mise en œuvre de la programmation simultanée par les systèmes d'exploitation modernes. L'utilisation de threads évite le gaspillage du cycle du processeur et augmente l'efficacité d'une application.
Jusqu'à présent, nous avons écrit les programmes où un seul thread s'exécute comme un seul processus qui est l'instance en cours d'exécution de l'application. Cependant, de cette façon, l'application peut effectuer une tâche à la fois. Pour qu'il exécute plus d'une tâche à la fois, il peut être divisé en threads plus petits.
Cycle de vie du fil
Le cycle de vie d'un thread commence lorsqu'un objet de la classe System.Threading.Thread est créé et se termine lorsque le thread est terminé ou termine l'exécution.
Voici les différents états du cycle de vie d'un thread -
The Unstarted State - Il s'agit du cas où l'instance du thread est créée mais que la méthode Start n'est pas appelée.
The Ready State - C'est la situation où le thread est prêt à s'exécuter et en attente du cycle CPU.
The Not Runnable State - Un thread n'est pas exécutable, quand
- La méthode de sommeil a été appelée
- La méthode d'attente a été appelée
- Bloqué par les opérations d'E / S
The Dead State - C'est la situation lorsque le thread termine son exécution ou est abandonné.
Le fil conducteur
En C #, le System.Threading.Threadclass est utilisé pour travailler avec les threads. Il permet de créer et d'accéder à des threads individuels dans une application multithread. Le premier thread à être exécuté dans un processus est appelé lemain fil.
Lorsqu'un programme C # démarre l'exécution, le thread principal est automatiquement créé. Les fils créés à l'aide duThreadclass sont appelés les threads enfants du thread principal. Vous pouvez accéder à un fil en utilisant leCurrentThread propriété de la classe Thread.
Le programme suivant montre l'exécution du thread principal -
using System;
using System.Threading;
namespace MultithreadingApplication {
class MainThreadProgram {
static void Main(string[] args) {
Thread th = Thread.CurrentThread;
th.Name = "MainThread";
Console.WriteLine("This is {0}", th.Name);
Console.ReadKey();
}
}
}
Lorsque le code ci-dessus est compilé et exécuté, il produit le résultat suivant -
This is MainThread
Propriétés et méthodes de la classe Thread
Le tableau suivant montre certains des properties du Thread classe -
N ° Sr. | Propriété et description |
---|---|
1 | CurrentContext Obtient le contexte actuel dans lequel le thread s'exécute. |
2 | CurrentCulture Obtient ou définit la culture du thread actuel. |
3 | CurrentPrinciple Obtient ou définit le principal actuel du thread (pour la sécurité basée sur les rôles). |
4 | CurrentThread Obtient le thread en cours d'exécution. |
5 | CurrentUICulture Obtient ou définit la culture actuelle utilisée par Resource Manager pour rechercher des ressources spécifiques à la culture au moment de l'exécution. |
6 | ExecutionContext Obtient un objet ExecutionContext qui contient des informations sur les différents contextes du thread actuel. |
sept | IsAlive Obtient une valeur indiquant l'état d'exécution du thread actuel. |
8 | IsBackground Obtient ou définit une valeur indiquant si un thread est ou non un thread d'arrière-plan. |
9 | IsThreadPoolThread Obtient une valeur indiquant si un thread appartient ou non au pool de threads managés. |
dix | ManagedThreadId Obtient un identificateur unique pour le thread géré actuel. |
11 | Name Obtient ou définit le nom du thread. |
12 | Priority Obtient ou définit une valeur indiquant la priorité de planification d'un thread. |
13 | ThreadState Obtient une valeur contenant les états du thread actuel. |
Le tableau suivant présente certains des methods du Thread classe -
N ° Sr. | Méthode et description |
---|---|
1 | public void Abort() Déclenche une ThreadAbortException dans le thread sur lequel il est appelé, pour commencer le processus de fin du thread. L'appel de cette méthode met généralement fin au thread. |
2 | public static LocalDataStoreSlot AllocateDataSlot() Alloue un emplacement de données sans nom sur tous les threads. Pour de meilleures performances, utilisez plutôt les champs marqués avec l'attribut ThreadStaticAttribute. |
3 | public static LocalDataStoreSlot AllocateNamedDataSlot(string name) Alloue un emplacement de données nommé sur tous les threads. Pour de meilleures performances, utilisez plutôt les champs marqués avec l'attribut ThreadStaticAttribute. |
4 | public static void BeginCriticalRegion() Avertit un hôte que l'exécution est sur le point d'entrer dans une région de code dans laquelle les effets d'une interruption de thread ou d'une exception non gérée peuvent compromettre d'autres tâches du domaine d'application. |
5 | public static void BeginThreadAffinity() Avertit un hôte que le code géré est sur le point d'exécuter des instructions qui dépendent de l'identité du thread du système d'exploitation physique actuel. |
6 | public static void EndCriticalRegion() Avertit un hôte que l'exécution est sur le point d'entrer une région de code dans laquelle les effets d'une interruption de thread ou d'une exception non gérée sont limités à la tâche en cours. |
sept | public static void EndThreadAffinity() Avertit un hôte que le code managé a terminé d'exécuter les instructions qui dépendent de l'identité du thread du système d'exploitation physique actuel. |
8 | public static void FreeNamedDataSlot(string name) Élimine l'association entre un nom et un emplacement, pour tous les threads du processus. Pour de meilleures performances, utilisez plutôt les champs marqués avec l'attribut ThreadStaticAttribute. |
9 | public static Object GetData(LocalDataStoreSlot slot) Récupère la valeur de l'emplacement spécifié sur le thread actuel, dans le domaine actuel du thread actuel. Pour de meilleures performances, utilisez plutôt les champs marqués avec l'attribut ThreadStaticAttribute. |
dix | public static AppDomain GetDomain() Renvoie le domaine actuel dans lequel le thread actuel s'exécute. |
11 | public static AppDomain GetDomainID() Renvoie un identifiant de domaine d'application unique |
12 | public static LocalDataStoreSlot GetNamedDataSlot(string name) Recherche un emplacement de données nommé. Pour de meilleures performances, utilisez plutôt les champs marqués avec l'attribut ThreadStaticAttribute. |
13 | public void Interrupt() Interrompt un thread qui est dans l'état de thread WaitSleepJoin. |
14 | public void Join() Bloque le thread appelant jusqu'à ce qu'un thread se termine, tout en continuant à effectuer le pompage COM et SendMessage standard. Cette méthode a différentes formes surchargées. |
15 | public static void MemoryBarrier() Synchronise l'accès à la mémoire comme suit: Le processeur exécutant le thread actuel ne peut pas réorganiser les instructions de telle sorte que les accès mémoire avant l'appel à MemoryBarrier s'exécutent après les accès mémoire qui suivent l'appel à MemoryBarrier. |
16 | public static void ResetAbort() Annule un abandon demandé pour le thread actuel. |
17 | public static void SetData(LocalDataStoreSlot slot, Object data) Définit les données dans l'emplacement spécifié sur le thread en cours d'exécution, pour le domaine actuel de ce thread. Pour de meilleures performances, utilisez plutôt les champs marqués avec l'attribut ThreadStaticAttribute. |
18 | public void Start() Démarre un fil. |
19 | public static void Sleep(int millisecondsTimeout) Met le thread en pause pendant un certain temps. |
20 | public static void SpinWait(int iterations) Fait attendre un thread le nombre de fois défini par le paramètre itérations |
21 | public static byte VolatileRead(ref byte address) public static double VolatileRead(ref double address) public static int VolatileRead(ref int address) public static Object VolatileRead(ref Object address) Lit la valeur d'un champ. La valeur est la dernière écrite par un processeur sur un ordinateur, quel que soit le nombre de processeurs ou l'état du cache du processeur. Cette méthode a différentes formes surchargées. Seuls quelques-uns sont donnés ci-dessus. |
22 | public static void VolatileWrite(ref byte address,byte value) public static void VolatileWrite(ref double address, double value) public static void VolatileWrite(ref int address, int value) public static void VolatileWrite(ref Object address, Object value) Écrit immédiatement une valeur dans un champ, de sorte que la valeur soit visible par tous les processeurs de l'ordinateur. Cette méthode a différentes formes surchargées. Seuls quelques-uns sont donnés ci-dessus. |
23 | public static bool Yield() Provoque le thread appelant pour céder l'exécution à un autre thread qui est prêt à s'exécuter sur le processeur actuel. Le système d'exploitation sélectionne le thread auquel céder. |
Créer des threads
Les threads sont créés en étendant la classe Thread. La classe Thread étendue appelle ensuite leStart() méthode pour commencer l'exécution du thread enfant.
Le programme suivant illustre le concept -
using System;
using System.Threading;
namespace MultithreadingApplication {
class ThreadCreationProgram {
public static void CallToChildThread() {
Console.WriteLine("Child thread starts");
}
static void Main(string[] args) {
ThreadStart childref = new ThreadStart(CallToChildThread);
Console.WriteLine("In Main: Creating the Child thread");
Thread childThread = new Thread(childref);
childThread.Start();
Console.ReadKey();
}
}
}
Lorsque le code ci-dessus est compilé et exécuté, il produit le résultat suivant -
In Main: Creating the Child thread
Child thread starts
Gérer les threads
La classe Thread fournit diverses méthodes de gestion des threads.
L'exemple suivant illustre l'utilisation du sleep() méthode pour faire une pause de thread pendant une période de temps spécifique.
using System;
using System.Threading;
namespace MultithreadingApplication {
class ThreadCreationProgram {
public static void CallToChildThread() {
Console.WriteLine("Child thread starts");
// the thread is paused for 5000 milliseconds
int sleepfor = 5000;
Console.WriteLine("Child Thread Paused for {0} seconds", sleepfor / 1000);
Thread.Sleep(sleepfor);
Console.WriteLine("Child thread resumes");
}
static void Main(string[] args) {
ThreadStart childref = new ThreadStart(CallToChildThread);
Console.WriteLine("In Main: Creating the Child thread");
Thread childThread = new Thread(childref);
childThread.Start();
Console.ReadKey();
}
}
}
Lorsque le code ci-dessus est compilé et exécuté, il produit le résultat suivant -
In Main: Creating the Child thread
Child thread starts
Child Thread Paused for 5 seconds
Child thread resumes
Détruire les fils
le Abort() est utilisée pour détruire les threads.
Le runtime abandonne le thread en lançant un ThreadAbortException. Cette exception ne peut pas être interceptée, le contrôle est envoyé au bloc finally , le cas échéant.
Le programme suivant illustre cela -
using System;
using System.Threading;
namespace MultithreadingApplication {
class ThreadCreationProgram {
public static void CallToChildThread() {
try {
Console.WriteLine("Child thread starts");
// do some work, like counting to 10
for (int counter = 0; counter <= 10; counter++) {
Thread.Sleep(500);
Console.WriteLine(counter);
}
Console.WriteLine("Child Thread Completed");
} catch (ThreadAbortException e) {
Console.WriteLine("Thread Abort Exception");
} finally {
Console.WriteLine("Couldn't catch the Thread Exception");
}
}
static void Main(string[] args) {
ThreadStart childref = new ThreadStart(CallToChildThread);
Console.WriteLine("In Main: Creating the Child thread");
Thread childThread = new Thread(childref);
childThread.Start();
//stop the main thread for some time
Thread.Sleep(2000);
//now abort the child
Console.WriteLine("In Main: Aborting the Child thread");
childThread.Abort();
Console.ReadKey();
}
}
}
Lorsque le code ci-dessus est compilé et exécuté, il produit le résultat suivant -
In Main: Creating the Child thread
Child thread starts
0
1
2
In Main: Aborting the Child thread
Thread Abort Exception
Couldn't catch the Thread Exception