Исходный код вики Некоторые моменты
Редактировал(а) Alexandr Fokin 2024/02/26 13:28
Последние авторы
| author | version | line-number | content |
|---|---|---|---|
| 1 | |(% style="width:181px" %)[[Исключения ~~| Exception >>doc:Разработка.NET.C#.Исключения | Exception .WebHome]]|(% style="width:1256px" %)((( | ||
| 2 | |(% style="width:181px" %)async Task ActionAsync|(% style="width:1026px" %)Если внутри метода возникнет исключение, то будет возвращен Task с исключением | ||
| 3 | |(% style="width:181px" %)Task ActionAsync|(% style="width:1026px" %)((( | ||
| 4 | Если внутри метода возникнет исключение, то оно будет проброшено наверх напрямую. | ||
| 5 | |||
| 6 | Если мы хотим сохранить общее поведение, то необходимо использовать конструкцию: | ||
| 7 | {{code language="C#"}}try | ||
| 8 | { | ||
| 9 | return Task.CompletedTask; | ||
| 10 | } | ||
| 11 | catch (Exception ex) | ||
| 12 | { | ||
| 13 | return Task.FromException(ex); | ||
| 14 | }{{/code}} | ||
| 15 | ))) | ||
| 16 | ))) | ||
| 17 | |(% style="width:181px" %)AsyncLocal|(% style="width:1256px" %)((( | ||
| 18 | |(% style="width:196px" %)async Task ActionAsync|(% style="width:1011px" %)В случае, если внутри метода будет задано значение AsyncLocal контейнера, то при выходе из метода оно будет сброшено на родительское значение. | ||
| 19 | |(% style="width:196px" %)Task ActionAsync|(% style="width:1011px" %)В случае, если внутри метода будет задано значение AsyncLocal контейнера, то при выходе из метода оно не будет сброшено. | ||
| 20 | ))) | ||
| 21 | |(% style="width:181px" %)Выполнение кода|(% style="width:1256px" %)((( | ||
| 22 | |(% style="width:48px" %)1)|(% style="width:456px" %){{code language="c#"}}var tasks = Enumerable | ||
| 23 | .Range(0, 10) | ||
| 24 | .Select( | ||
| 25 | async e => { | ||
| 26 | action1(); | ||
| 27 | await Task.Delay(TimeSpan.FromSeconds(10)); | ||
| 28 | action2(); | ||
| 29 | } | ||
| 30 | ) | ||
| 31 | .ToArray(); | ||
| 32 | await Task.WhenAll(tasks);{{/code}}|(% style="width:776px" %)action1 будут выполнены одним потоком (который вызвал Enumarable) последовательно друг за другом (без какого-либо параллелизма). После этого происходит асинхронное ожидание и прерывание потока исполнения. | ||
| 33 | action2 будет выполнено после ожидания при этом поток его выполнения зависит от параметров [[SynchronizationContext ~~| Контекст синхронизации>>doc:Разработка.NET.C#.Многопоточность и асинхронность.Асинхронность\. Async Await.SynchronizationContext | Контекст синхронизации.WebHome]] окружения (при этом могут исполняться параллельно друг другу в разных потоках. | ||
| 34 | \\Такой подход подходит для ситуации, когда мы не стремимся параллелить action1 (он не является ресурсоемким). | ||
| 35 | |(% style="width:48px" %)2)|(% style="width:456px" %){{code language="c#"}}var tasks = Enumerable | ||
| 36 | .Range(0, 10) | ||
| 37 | .Select( | ||
| 38 | e => Task.Run( | ||
| 39 | async () => { | ||
| 40 | action1(); | ||
| 41 | await Task.Delay(TimeSpan.FromSeconds(10)); | ||
| 42 | action2(); | ||
| 43 | } | ||
| 44 | ) | ||
| 45 | ) | ||
| 46 | .ToArray(); | ||
| 47 | await Task.WhenAll(tasks);{{/code}}|(% style="width:776px" %)При таком вызове на каждый отдельный элемент будет запущена отдельная задача через планировщик. Действия будут выполняться параллельно друг другу (в том числе и action1) в разных потоках. Будет запланировано 10 задач. | ||
| 48 | По умолчанию код, внутри делегата Task.Run (или Parallel.ForEachAsync) будет выполнен без контекста синхронизации, таким образом action2 будет запланирован к исполнению на потоке их пула. | ||
| 49 | \\Такой подход не очень хорош, если каждая операция не является ресурсоемкой с точки зрения процессора. | ||
| 50 | |(% style="width:48px" %)3) |(% style="width:456px" %){{code language="c#"}}await Parallel.ForEachAsync( | ||
| 51 | Enumerable.Range(0, 10), | ||
| 52 | async (e, t) => | ||
| 53 | { | ||
| 54 | action1(); | ||
| 55 | await Task.Delay(TimeSpan.FromSeconds(10)); | ||
| 56 | action2(); | ||
| 57 | } | ||
| 58 | );{{/code}}|(% style="width:776px" %)Данный метод позволяет выполнять действия параллельно, но при этом будет более экономичным чем 2 вариант т.к. | ||
| 59 | |||
| 60 | |||
| 61 | ))) | ||
| 62 | |||
| 63 |