Форум программистов, компьютерный форум, киберфорум
C# для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.87/15: Рейтинг темы: голосов - 15, средняя оценка - 4.87
 Аватар для SunRain
13 / 13 / 2
Регистрация: 03.09.2012
Сообщений: 133

Почему async/await выполняются синхронно?

07.11.2016, 10:59. Показов 2910. Ответов 12
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Добрый день, Гуру!
Немного запутался в логике работы async/await, очень прошу просветить..)
Есть несколько асинхронных методов, выполняющихся синхронно в одном таске, хотелось бы понять почему.

Как я это понимаю:
-В строке 1 синхронно запускается метод MethodCaller()
-В строке 2 MethodCaller() возвращает в Main() незавершенный Task и запускает MethodA() в новом потоке, в то время как основной поток выполняет строку 3.

Но все выполняется полностью синхронно, т.е. строка 3 выполняется последней. Почему?

P.S. И еще один вопрос: асинхронные методы идут каскадом, скажем, десяток асинхронных методов дергает один другой. При каждом встреченном await из пула будет извлекаться поток?

C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
 
 
        private static void Main()
        {
            MethodCaller(); //1
            Console.WriteLine($"Task - {Task.CurrentId}: Main do work"); //3
            Console.ReadKey();
        }
 
        private static async Task MethodCaller()
        {
            Console.WriteLine($"Task - {Task.CurrentId}: Start MethodCaller()");
            var myNum = await MethodA(); //2
            Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodCaller(), {myNum}");
        }
 
        private static async Task<int> MethodA()
        {
            Console.WriteLine($"Task - {Task.CurrentId}: Start MethodA()");
            var result = await MethodB(); 
            Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodA()");
            return result;
        }
 
        private static async Task<int> MethodB()
        {
            Console.WriteLine($"Task - {Task.CurrentId}: Start MethodB()");
            var randomNumber = new Random().Next(5, 10);
            Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodB()");
            return randomNumber;
        }
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
07.11.2016, 10:59
Ответы с готовыми решениями:

Async/ await как правильно ввести данные в async метод (консоль)
Привет , кто то может помочь ?) проблема в тому что у меня есть async метод который запускается из Main, по среди этого метода...

async/await
https://metanit.com/sharp/tutorial/13.7.php вот код Task&lt;T&gt;: // определение асинхронного метода static async...

async/await (._. )
Опять я с глупым вопросом. Не могу понять, почему метод, который выгружает данные не асинхронный? Вообще форма зависает на неопределенное...

12
Эксперт .NET
 Аватар для Usaga
14298 / 9383 / 1353
Регистрация: 21.01.2016
Сообщений: 35,374
07.11.2016, 12:19
SunRain, так у тебя метод MethodB синхронный, потому и вся цепочка работает синхронно.
1
 Аватар для SunRain
13 / 13 / 2
Регистрация: 03.09.2012
Сообщений: 133
07.11.2016, 12:59  [ТС]
То есть, если в цепочке хотя бы один метод выполняется синхронно, вся цепочка становиться синхронной?

Добавлено через 6 минут
Как должен выглядеть MethodB(), что бы код был асинхронным?
0
 Аватар для LeniumSoft
1454 / 847 / 150
Регистрация: 06.06.2012
Сообщений: 2,370
07.11.2016, 13:03
Цитата Сообщение от SunRain Посмотреть сообщение
То есть, если в цепочке хотя бы один метод выполняется синхронно, вся цепочка становиться синхронной?
Выполни этот код в WinForms приложении. Всё будет асинхронным.
async/await в консоли нормально не пашут.

Добавлено через 2 минуты
Console можно заменить на Debug и смотреть в окно Output
1
 Аватар для SunRain
13 / 13 / 2
Регистрация: 03.09.2012
Сообщений: 133
07.11.2016, 13:05  [ТС]
LeniumSoft,
Да, я знаю, в WPF все у меня отлично отрабатывает, поэтому и хочу понять в чем проблема..)
Причем, если в консоли вместо самодельных MethodA и MethodB ждать просто Task.Delay(1000), то все отработает асинхронно, поэтому и хочется понять, что именно я делаю не так..(
0
 Аватар для EveKS
601 / 485 / 185
Регистрация: 19.04.2016
Сообщений: 1,885
07.11.2016, 13:31
Возможно, можно это:
C#
1
2
3
4
5
6
7
        private static async Task<int> MethodB()
        {
            Console.WriteLine($"Task - {Task.CurrentId}: Start MethodB()");
            var randomNumber = new Random().Next(5, 10);
            Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodB()");
            return randomNumber;
        }
Заменить на
C#
1
2
3
4
5
6
7
8
        private static async Task<int> MethodB()
=>await Task.Run(()=>
        {
            Console.WriteLine($"Task - {Task.CurrentId}: Start MethodB()");
            var randomNumber = new Random().Next(5, 10);
            Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodB()");
            return randomNumber;
        });
1
 Аватар для SunRain
13 / 13 / 2
Регистрация: 03.09.2012
Сообщений: 133
07.11.2016, 14:06  [ТС]
EveKS,
Да, так асинхронность заработала, вопрос насколько good practice считается подобный код..

Кстати, почему метод MethodA() после await запускается в том же таске, что и MethodCaller()?
Получается до первого явного запуска нового таска (как в MethodB()) реально все будет выполнятся синхронно?
0
22 / 20 / 3
Регистрация: 12.10.2016
Сообщений: 62
07.11.2016, 14:23
SunRain, рекомендую изучить это

Добавлено через 5 минут
И это
1
 Аватар для SunRain
13 / 13 / 2
Регистрация: 03.09.2012
Сообщений: 133
07.11.2016, 14:32  [ТС]
gvt,
Спасибо, отличная статья..)
Получается аналог асинхронных методов самого .NET, которые выполняют какую-то ресурсоемкую работу стоит делать как раз с помощью TaskCompletionSource, а уже их вызов осуществлять каскадом async/await методов?

Что-то вроде

C#
1
2
3
4
5
6
7
8
9
10
11
12
 private static Task<int> MethodBAsync()
        {
            var task = Task.Factory.StartNew(() =>
            {
                Console.WriteLine($"Task - {Task.CurrentId}: Start MethodB()");
                var randomNumber = new Random().Next(5, 10);
                Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodB()");
                return randomNumber;
            });
            var tcs = new TaskCompletionSource<int>(task);
            return tcs.Task;
        }
0
 Аватар для IamRain
4694 / 2702 / 734
Регистрация: 02.08.2011
Сообщений: 7,233
07.11.2016, 14:55
Цитата Сообщение от SunRain Посмотреть сообщение
стоит делать как раз с помощью TaskCompletionSource
Необязательно. Акцент в статье был на то, чтобы реализовывать асинхронность, затрачивая на это меньше ресурсов, а не просто оборачивая синхронные методы в Task-и. Читайте внимательнее.

Ятп в вашем случае, TaskCompletionSource нафиг не нужен. Вы ведь создали и запустили Task - для чего вам нужен этот TaskCompletionSource?

C#
1
2
3
4
5
6
7
8
9
10
private static Task<int> MethodBAsync()
        {
           return  Task.Factory.StartNew(() =>
            {
                Console.WriteLine($"Task - {Task.CurrentId}: Start MethodB()");
                var randomNumber = new Random().Next(5, 10);
                Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodB()");
                return randomNumber;
            });            
        }
0
 Аватар для EveKS
601 / 485 / 185
Регистрация: 19.04.2016
Сообщений: 1,885
07.11.2016, 15:19
Я особо не понимаю в тасках, вот чуть переделал написанное выше:
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
using System;
using System.Threading;
using System.Threading.Tasks;
 
class Program
{
    private static void Main()
    {
        CancellationTokenSource cancelTokSrc = new CancellationTokenSource();
        Task tsk = Task.Factory.StartNew(MethodCaller, cancelTokSrc.Token, cancelTokSrc.Token); //1
        Console.WriteLine($"Task - {Task.CurrentId}: Main do work"); //3
        tsk.GetAwaiter().GetResult();
 
        Console.ReadKey();
    }
 
    private static async Task MethodCaller(object cancellationToken)
    {
        Console.WriteLine($"Task - {Task.CurrentId}: Start MethodCaller()");
        var myNum = await MethodA((CancellationToken)cancellationToken); //2
        Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodCaller(), {myNum}");
    }
 
    private static async Task<int> MethodA(CancellationToken cancellationToken)
    {
        Console.WriteLine($"Task - {Task.CurrentId}: Start MethodA()");
        var result = await MethodBAsync(cancellationToken);
        Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodA()");
        return result;
    }
    public static async Task<int> MethodBAsync(CancellationToken cancellationToken)
    => await Task.Run(() =>
        {
            Console.WriteLine($"Task - {Task.CurrentId}: Start MethodB()");
            var randomNumber = new Random().Next(5, 10);
            Console.WriteLine($"Task - {Task.CurrentId}: Finish MethodB()");
            return randomNumber;
        }, cancellationToken);
}
Теперь можно отменять задачу через cancelTokSrc.Cancel();
0
 Аватар для IamRain
4694 / 2702 / 734
Регистрация: 02.08.2011
Сообщений: 7,233
07.11.2016, 15:28
EveKS,
насколько я помню, чтобы отменить таску, нужно вбросить исключение, у CancellationToken есть для этого специализированный метод Throwifcancellationrequested.
Если таска уже отменена и при этом не успела запуститься - она и не запустится. Если же уже запущена - то сама логика в таске должна явно поддерживать операцию отмены. (С) Рихтер.
И единственный способ прервать выполнение и уведомить клиентский код об отмене - вбросить исключение.
0
 Аватар для EveKS
601 / 485 / 185
Регистрация: 19.04.2016
Сообщений: 1,885
07.11.2016, 15:48
IamRain,
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
        CancellationTokenSource cancelTokSrc = new CancellationTokenSource();
        Task tsk = Task.Factory.StartNew(MethodCaller, cancelTokSrc.Token, cancelTokSrc.Token); //1
        try
        {
            cancelTokSrc.Cancel();
            tsk.GetAwaiter().GetResult();
        }
        catch(AggregateException aex)
        {
           
        }
        catch (Exception ex)
        {
            // todo:
        }
        finally
        {
            tsk.Dispose();
            cancelTokSrc.Dispose();
        }
C#
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
        CancellationTokenSource cancelTokSrc = new CancellationTokenSource();
        Task tsk = Task.Factory.StartNew(MethodCaller, cancelTokSrc.Token, cancelTokSrc.Token); //1
        try
        {
            cancelTokSrc.Cancel();
            tsk.Wait();
        }
        catch(AggregateException aex)
        {
            // todo:
        }
        catch (Exception ex)
        {
 
        }
        finally
        {
            tsk.Dispose();
            cancelTokSrc.Dispose();
        }
1
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
07.11.2016, 15:48
Помогаю со студенческими работами здесь

Async/await
В интернете копался ничего информативного не нашел, все в каких- то не понятных для новичка терминах, объясните пожалуйста смысл...

Async await
Пытыюсь разобраться с async/await но что то без успешно пока. Не подскажете как переделать этот код на примере проще освоить public...

Task и async-await
Объясните пожалуйста, в чем преимущество async-await? И в чем разница между обычной задачей? Ведь задачи то же выполняются независимо друг...

Работа с async await
Добрый вечер. Помогите исправить код. Мне нужно сделать поиск по реестру, например ввожу слово Console, программа должна получить все имена...

Async await + zeromq
Добрый день, уважаемые гуру. Помогите разобраться с проблемой уже несколько недель бьюсь. Имеется клиент-сервер на zeromq Клиент: ...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
13
Ответ Создать тему
Новые блоги и статьи
Модульная разработка через nuget packages
DevAlt 07.03.2026
Сложившийся в . Net-среде способ разработки чаще всего предполагает монорепозиторий в котором находятся все исходники. При создании нового решения, мы просто добавляем нужные проекты и имеем. . .
Модульный подход на примере F#
DevAlt 06.03.2026
В блоге дяди Боба наткнулся на такое определение: В этой книге («Подход, основанный на вариантах использования») Ивар утверждает, что архитектура программного обеспечения — это структуры,. . .
Управление камерой с помощью скрипта OrbitControls.js на Three.js: Вращение, зум и панорамирование
8Observer8 05.03.2026
Содержание блога Финальная демка в браузере работает на Desktop и мобильных браузерах. Итоговый код: orbit-controls-threejs-js. zip. Сканируйте QR-код на мобильном. Вращайте камеру одним пальцем,. . .
SDL3 для Web (WebAssembly): Синхронизация спрайтов SDL3 и тел Box2D
8Observer8 04.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-sync-physics-sprites-sdl3-c. zip На первой гифке отладочные линии отключены, а на второй включены:. . .
SDL3 для Web (WebAssembly): Идентификация объектов на Box2D v3 - использование userData и событий коллизий
8Observer8 02.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-collision-events-sdl3-c. zip Сканируйте QR-код на мобильном и вы увидите, что появится джойстик для управления главным героем. . . .
Реалии
Hrethgir 01.03.2026
Нет, я не закончил до сих пор симулятор. Эта задача сложнее. Не получилось уйти в плавсостав, но оно и к лучшему, возможно. Точнее получалось - но сварщиком в палубную команду, а это значит, в моём. . .
Ритм жизни
kumehtar 27.02.2026
Иногда приходится жить в ритме, где дел становится всё больше, а вовлечения в происходящее — всё меньше. Плотный график не даёт вниманию закрепиться ни на одном событии. Утро начинается с быстрых,. . .
SDL3 для Web (WebAssembly): Сборка библиотек: SDL3, Box2D, FreeType, SDL3_ttf, SDL3_mixer и SDL3_image из исходников с помощью CMake и Emscripten
8Observer8 27.02.2026
Недавно вышла версия 3. 4. 2 библиотеки SDL3. На странице официальной релиза доступны исходники, готовые DLL (для x86, x64, arm64), а также библиотеки для разработки под Android, MinGW и Visual Studio. . . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru