Maison  >  Article  >  développement back-end  >  Explication détaillée du code graphique de Thread, Task, Async/Await, IAsyncResult en C#

Explication détaillée du code graphique de Thread, Task, Async/Await, IAsyncResult en C#

黄舟
黄舟original
2017-03-29 11:28:103239parcourir

Cet article présente principalement les connaissances pertinentes de Thread, Task, Async/Await, IAsyncResult en C#. Il a une certaine valeur de référence, jetons un oeil avec l'éditeur ci-dessous

En parlant d'asynchrone, Thread, Task, async/await, IAsyncResult, ces choses sont définitivement incontournables. Parlons-en tour à tour aujourd'hui

1. Thread

L'importance du multi-threading est que dans une application, plusieurs parties d'exécution peuvent être exécutées en même temps pour des opérations plus chronophages (pour exemple, io,

httpopérations de base de données ), ou en attente d'une réponse (telle comme communication WCF) L'opération peut être effectuée en démarrant le thread d'arrière-plan séparément, afin que le thread principal ne soit pas bloqué et puisse continuer à s'exécuter en attendant que le thread d'arrière-plan soit terminé, puis en informant le thread principal, puis en effectuant le correspondant ; opération

en C# ! Il est relativement simple de démarrer un nouveau thread en

static void Main(string[] args)
{
 Console.WriteLine("主线程开始");
 //IsBackground=true,将其设置为后台线程
 Thread t = new Thread(Run) { IsBackground = true };
 t.Start();
   Console.WriteLine("主线程在做其他的事!");
 //主线程结束,后台线程会自动结束,不管有没有执行完成
 //Thread.Sleep(300);
 Thread.Sleep(1500);
 Console.WriteLine("主线程结束");
}
static void Run()
{
 Thread.Sleep(700);
 Console.WriteLine("这是后台线程调用");
}
Le résultat de l'exécution est comme indiqué ci-dessous,

Vous pouvez voir qu'après avoir démarré le thread d'arrière-plan, le thread principal continue de s'exécuter

1.1 Thread Pool

Imaginez s'il y a un un grand nombre de tâches à traiter, telles que Pour le traitement des requêtes HTTP en arrière-plan du site web, est-il nécessaire de créer un thread en arrière-plan pour chaque requête ? C'est évidemment inapproprié. Cela prendra beaucoup de mémoire, et. le processus de création fréquent affectera également sérieusement la vitesse. Alors, que dois-je faire ? Le but du pool est de résoudre ce problème. Il stocke les threads créés pour former un pool de threads (avec plusieurs threads). à traiter, s'il y a des threads inactifs dans le pool de threads (une fois l'exécution de la tâche précédente terminée, le thread ne sera pas recyclé et mis à l'état inactif), puis appelez directement l'exécution du thread dans le pool de threads ( par exemple, l'objet Application

dans le mécanisme de traitement asp.net),

Exemple d'utilisation :

for (int i = 0; i < 10; i++)
{
 ThreadPool.QueueUserWorkItem(m =>
 {
  Console.WriteLine(Thread.CurrentThread.ManagedThreadId.ToString());
 });
}
Console.Read();
Résultat d'exécution :

Vous pouvez voir que bien qu'il ait été exécuté 10 fois, 10 threads n'ont pas été créés

1.2 Sémaphore .

Sémaphore est responsable de la coordination des threads et peut limiter le nombre de threads accédant à une certaine ressource

Voici un exemple simple d'utilisation de la classe SemaphoreSlim :

 static SemaphoreSlim semLim = new SemaphoreSlim(3); //3表示最多只能有三个线程同时访问
static void Main(string[] args)
{
 for (int i = 0; i < 10; i++)
 {
  new Thread(SemaphoreTest).Start();
 }
 Console.Read();
}
static void SemaphoreTest()
{
 semLim.Wait();
 Console.WriteLine("线程" + Thread.CurrentThread.ManagedThreadId.ToString() + "开始执行");
 Thread.Sleep(2000);
 Console.WriteLine("线程" + Thread.CurrentThread.ManagedThreadId.ToString() + "执行完毕");
 semLim.Release();
}
Les résultats de l'exécution sont les suivants :

Comme vous pouvez le voir, au début, il n'y a que trois threads en cours d'exécution. Lorsqu'un thread est exécuté et libéré, il y a. sera un nouveau fil pour exécuter la méthode.

En plus de la classe SemaphoreSlim, vous pouvez également utiliser la classe Semaphore, qui semble plus flexible. Si vous êtes intéressé, vous pouvez la rechercher, il y en aura ! pas de démonstration ici !

2.Task

La tâche a été ajoutée dans .NET4.0. Elle a des fonctions similaires au pool de threads ThreadPool. Lorsque vous utilisez Task pour démarrer une nouvelle tâche, il sera tiré du pool de threads appelant et Thread créera un nouveau thread à chaque fois qu'il sera instancié.

Console.WriteLine("主线程启动");
//Task.Run启动一个线程
//Task启动的是后台线程,要在主线程中等待后台线程执行完毕,可以调用Wait方法
//Task task = Task.Factory.StartNew(() => { Thread.Sleep(1500); Console.WriteLine("task启动"); });
Task task = Task.Run(() => { 
 Thread.Sleep(1500);
 Console.WriteLine("task启动");
});
Thread.Sleep(300);
task.Wait();
Console.WriteLine("主线程结束");
Les résultats de l'exécution sont les suivants :

Méthode pour démarrer une nouvelle tâche : Task.Run() ou Task.Factory.StartNew( ), open C'est un thread d'arrière-plan

Pour attendre que le thread d'arrière-plan termine son exécution dans le thread principal, vous pouvez utiliser la méthode Wait (il sera exécuté de manière synchrone). Sans Wait, il sera exécuté de manière asynchrone.

Comparez la tâche et le fil de discussion :

static void Main(string[] args)
{
 for (int i = 0; i < 5; i++)
 {
  new Thread(Run1).Start();
 }
 for (int i = 0; i < 5; i++)
 {
  Task.Run(() => { Run2(); });
 }
}
static void Run1()
{
 Console.WriteLine("Thread Id =" + Thread.CurrentThread.ManagedThreadId);
}
static void Run2()
{
 Console.WriteLine("Task调用的Thread Id =" + Thread.CurrentThread.ManagedThreadId);
}
Résultat de l'exécution :

On peut voir que l'utilisation directe de Thread ouvrira 5 fils de discussion , 3 ouverts à l'aide de Task (en utilisant le pool de threads) !

2.1 Taskb54c2c292509147c0b54128f4eb90887

Taskb54c2c292509147c0b54128f4eb90887 est une tâche avec une valeur de retour et TResult est le type de valeur de retour.

Console.WriteLine("主线程开始");
//返回值类型为string
Task<string> task = Task<string>.Run(() => {
 Thread.Sleep(2000); 
 return Thread.CurrentThread.ManagedThreadId.ToString(); 
});
//会等到task执行完毕才会输出;
Console.WriteLine(task.Result);
Console.WriteLine("主线程结束");
Résultat de l'exécution :

La valeur de retour peut être obtenue via task.Result Si le thread d'arrière-plan n'a pas terminé son exécution lorsque la valeur est. obtenu, il attendra que son exécution soit terminée !

Une brève mention :

Les tâches peuvent être annulées via la classe CancellationTokenSource. J'ai l'impression qu'elle n'est pas beaucoup utilisée et que l'utilisation est relativement simple. Si vous êtes intéressé, vous pouvez la rechercher. !

3. async/await

async/await a été introduit en C# 5.0, première utilisation :

static void Main(string[] args)
{
 Console.WriteLine("-------主线程启动-------");
 Task<int> task = GetStrLengthAsync();
 Console.WriteLine("主线程继续执行");
 Console.WriteLine("Task返回的值" + task.Result);
 Console.WriteLine("-------主线程结束-------");
}
static async Task<int> GetStrLengthAsync()
{
 Console.WriteLine("GetStrLengthAsync方法开始执行");
 //此处返回的<string>中的字符串类型,而不是Task<string>
 string str = await GetString();
 Console.WriteLine("GetStrLengthAsync方法执行结束");
 return str.Length;
}
static Task<string> GetString()
{
   //Console.WriteLine("GetString方法开始执行")
 return Task<string>.Run(() =>
 {
  Thread.Sleep(2000);
  return "GetString的返回值";
 });
}
async est utilisé Modifier la méthode pour indiquer que cette méthode est asynchrone. Le type de retour de la méthode déclarée doit être : void, Task ou Taskb54c2c292509147c0b54128f4eb90887.

await必须用来修饰Task或Taskb54c2c292509147c0b54128f4eb90887,而且只能出现在已经用async关键字修饰的异步方法中。通常情况下,async/await成对出现才有意义,

看看运行结果:

可以看出来,main函数调用GetStrLengthAsync方法后,在await之前,都是同步执行的,直到遇到await关键字,main函数才返回继续执行。

那么是否是在遇到await关键字的时候程序自动开启了一个后台线程去执行GetString方法呢?

现在把GetString方法中的那行注释加上,运行的结果是:

大家可以看到,在遇到await关键字后,没有继续执行GetStrLengthAsync方法后面的操作,也没有马上反回到main函数中,而是执行了GetString的第一行,以此可以判断await这里并没有开启新的线程去执行GetString方法,而是以同步的方式让GetString方法执行,等到执行到GetString方法中的Task98c455a79ddfebb79781bff588e7b37e.Run()的时候才由Task开启了后台线程!

那么await的作用是什么呢?

可以从字面上理解,上面提到task.wait可以让主线程等待后台线程执行完毕,await和wait类似,同样是等待,等待Task98c455a79ddfebb79781bff588e7b37e.Run()开始的后台线程执行完毕,不同的是await不会阻塞主线程,只会让GetStrLengthAsync方法暂停执行。

那么await是怎么做到的呢?有没有开启新线程去等待?

只有两个线程(主线程和Task开启的线程)!至于怎么做到的(我也不知道......>_03065e15d4ebf3426e4f9cf8ae830fea:

List<int> list = new List<int>() { 1, 2, 3, 4, 5, 6, 6, 7, 8, 9 };
Parallel.ForEach<int>(list, n =>
{
 Console.WriteLine(n);
 Thread.Sleep(1000);
});

执行Action[]数组里面的方法:

Action[] actions = new Action[] { 
 new Action(()=>{
  Console.WriteLine("方法1");
 }),
 new Action(()=>{
  Console.WriteLine("方法2");
 })
};
Parallel.Invoke(actions);

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn