Форум программистов, компьютерный форум, киберфорум
C# для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 4.67/15: Рейтинг темы: голосов - 15, средняя оценка - 4.67
24 / 10 / 5
Регистрация: 30.01.2015
Сообщений: 175

Переписать асинхронный метод без использования async/await

03.06.2016, 15:47. Показов 3076. Ответов 7
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Ломаю голову как переписать асинхронный метод что бы достичь того же эффекта как при использовании async/await.
К примеру(пример взят из учебника Скит Д. С# для профессионалов.Тонкости программирования.Третье издание)
C#
1
2
3
4
5
6
7
8
9
        private async void DisplayWebSiteLengthAsync(object sender, EventArgs e)
        {
            label.Text = "Fetching...";
            using (HttpClient client = new HttpClient())
            {
                string text = await client.GetStringAsync("https://habrahabr.ru/");
                label.Text = text.Length.ToString();
            }
        }
Как постигнуть такой же результат при исполнении но обычным методом. То есть без использования await.
К примеру у меня вышло что то такое:
C#
1
2
3
4
5
6
7
8
9
10
11
        private void DisplayWebSiteLength(object sender, EventArgs e)
        {
            label.Text = "Fetching...";
            using (WebClient client = new WebClient())
            {
                var text = Task.Run(() => client.DownloadString("https://habrahabr.ru/"))
                    .GetAwaiter();
 
                text.OnCompleted(() => label.Text = text.GetResult().Length.ToString());
            }
        }
Однако, хоть GetAwaiter() и возвращает сразу же управление работающему потоку, UI продолжит своё выполнение, то есть можно с легкостью написать MessadgeBox, и он его покажет не дожидаясь заврешения Task. Как подождать завершение работы Task, не блокируя UI поток, но при это и не продолжая его работы.
Собственно что и делает await.
0
IT_Exp
Эксперт
34794 / 4073 / 2104
Регистрация: 17.06.2006
Сообщений: 32,602
Блог
03.06.2016, 15:47
Ответы с готовыми решениями:

Асинхронный метод: async + await
Доброго времени суток! Есть достаточно простой метод: public UserControl1 ListToArray() { UserControl1 uc =...

Переписать код без использования async
Есть метод, нужно переписать его под обычный, чтобы программа подождала выполнения Идей 0..... string proxy =...

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

7
979 / 874 / 350
Регистрация: 26.04.2012
Сообщений: 2,647
03.06.2016, 16:45
А зачем оно, собственно, надо? Если брать WebClient и можно использовать сам класс Task, то можно как тут. Это заработает на .NET 4.0. А GetAwaiter-это тоже из .NET 4.5, если мне память не изменяет. Это используется компилятором в реализации того же async-await.
1
Master of Orion
Эксперт .NET
 Аватар для Psilon
6102 / 4958 / 905
Регистрация: 10.07.2011
Сообщений: 14,522
Записей в блоге: 5
03.06.2016, 18:11
kol,
C#
1
2
3
4
5
6
7
8
Task.Run(() =>
{
    Thread.Sleep(1000);
    using (WebClient client = new WebClient())
    {
        return client.DownloadString("https://habrahabr.ru/");
    }
}).ContinueWith(task => Invoke(new Action(() => MessageBox.Show("Total length is " + task.Result.Length))));
Добавлено через 1 минуту
Если точнее, то:
C#
1
2
3
4
5
6
7
8
9
var context = SynchronizationContext.Current;
Task.Run(() =>
{
    Thread.Sleep(1000);
    using (WebClient client = new WebClient())
    {
        return client.DownloadString("https://habrahabr.ru/");
    }
}).ContinueWith(task => context.Post(state => MessageBox.Show("Total length is " + state), task.Result.Length));
1
Эксперт .NET
 Аватар для insite2012
5548 / 4311 / 1218
Регистрация: 12.10.2013
Сообщений: 12,371
Записей в блоге: 2
03.06.2016, 18:50
Цитата Сообщение от EvilFromHell Посмотреть сообщение
GetAwaiter-это тоже из .NET 4.5, если мне память не изменяет.
Все верно. Однако фичи async-await вполне можно использовать и с NET4.0. Есть пакет для этого.
1
24 / 10 / 5
Регистрация: 30.01.2015
Сообщений: 175
03.06.2016, 19:06  [ТС]
EvilFromHell, Что ж как тогда написать используя метод используя GetAwaiter(). Зачем это надо? Разве плохо знать во что разворачивается ключевое слово await? Ментор попросил переписать, что бы мы более детально понимали принцип работы.
0
Master of Orion
Эксперт .NET
 Аватар для Psilon
6102 / 4958 / 905
Регистрация: 10.07.2011
Сообщений: 14,522
Записей в блоге: 5
03.06.2016, 19:18
kol, ну если хочется, знайте, во что разворачивается, на примере выше:
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
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
        // WindowsFormsApplication1.Form1
        [DebuggerStepThrough, AsyncStateMachine(typeof(Form1.<DisplayWebSiteLengthAsync>d__2))]
        private void DisplayWebSiteLengthAsync(object sender, EventArgs e)
        {
            Form1.<DisplayWebSiteLengthAsync>d__2 <DisplayWebSiteLengthAsync>d__ = new Form1.<DisplayWebSiteLengthAsync>d__2();
            <DisplayWebSiteLengthAsync>d__.<>4__this = this;
            <DisplayWebSiteLengthAsync>d__.sender = sender;
            <DisplayWebSiteLengthAsync>d__.e = e;
            <DisplayWebSiteLengthAsync>d__.<>t__builder = AsyncVoidMethodBuilder.Create();
            <DisplayWebSiteLengthAsync>d__.<>1__state = -1;
            AsyncVoidMethodBuilder <>t__builder = <DisplayWebSiteLengthAsync>d__.<>t__builder;
            <>t__builder.Start<Form1.<DisplayWebSiteLengthAsync>d__2>(ref <DisplayWebSiteLengthAsync>d__);
        }
 
        private sealed class <DisplayWebSiteLengthAsync>d__2 : IAsyncStateMachine
        {
            public int <>1__state;
 
            public AsyncVoidMethodBuilder <>t__builder;
 
            public object sender;
 
            public EventArgs e;
 
            public Form1 <>4__this;
 
            private HttpClient <client>5__1;
 
            private string <text>5__2;
 
            private string <>s__3;
 
            private TaskAwaiter<string> <>u__1;
 
            void IAsyncStateMachine.MoveNext()
            {
                int num = this.<>1__state;
                try
                {
                    if (num != 0)
                    {
                        this.<>4__this.label.Text = "Fetching...";
                        this.<client>5__1 = new HttpClient();
                    }
                    try
                    {
                        TaskAwaiter<string> taskAwaiter;
                        if (num != 0)
                        {
                            taskAwaiter = this.<client>5__1.GetStringAsync("https://habrahabr.ru/").GetAwaiter();
                            if (!taskAwaiter.IsCompleted)
                            {
                                num = (this.<>1__state = 0);
                                this.<>u__1 = taskAwaiter;
                                Form1.<DisplayWebSiteLengthAsync>d__2 <DisplayWebSiteLengthAsync>d__ = this;
                                this.<>t__builder.AwaitUnsafeOnCompleted<TaskAwaiter<string>, Form1.<DisplayWebSiteLengthAsync>d__2>(ref taskAwaiter, ref <DisplayWebSiteLengthAsync>d__);
                                return;
                            }
                        }
                        else
                        {
                            taskAwaiter = this.<>u__1;
                            this.<>u__1 = default(TaskAwaiter<string>);
                            num = (this.<>1__state = -1);
                        }
                        string result = taskAwaiter.GetResult();
                        taskAwaiter = default(TaskAwaiter<string>);
                        this.<>s__3 = result;
                        this.<text>5__2 = this.<>s__3;
                        this.<>s__3 = null;
                        this.<>4__this.label.Text = this.<text>5__2.Length.ToString();
                        this.<text>5__2 = null;
                    }
                    finally
                    {
                        if (num < 0 && this.<client>5__1 != null)
                        {
                            ((IDisposable)this.<client>5__1).Dispose();
                        }
                    }
                    this.<client>5__1 = null;
                }
                catch (Exception exception)
                {
                    this.<>1__state = -2;
                    this.<>t__builder.SetException(exception);
                    return;
                }
                this.<>1__state = -2;
                this.<>t__builder.SetResult();
            }
 
            [DebuggerHidden]
            void IAsyncStateMachine.SetStateMachine(IAsyncStateMachine stateMachine)
            {
            }
        }
0
Эксперт .NET
 Аватар для insite2012
5548 / 4311 / 1218
Регистрация: 12.10.2013
Сообщений: 12,371
Записей в блоге: 2
03.06.2016, 19:27
Цитата Сообщение от Psilon Посмотреть сообщение
знайте, во что разворачивается
Psilon, думаю, такие подробности излишни. Но вот знать, что такое признак ожидания (awaiter) и как его можно использовать, думаю, лишним точно не будет.
0
Master of Orion
Эксперт .NET
 Аватар для Psilon
6102 / 4958 / 905
Регистрация: 10.07.2011
Сообщений: 14,522
Записей в блоге: 5
03.06.2016, 20:21
insite2012, ну человек хотел понять, как оно работает Вот так оно и работает. Правда, еще куча кода скрыта в самом builder.Start.

kol, кстати насчет метода, см. мсдн:
https://msdn.microsoft.com/ru-... .110).aspx

This method is intended for compiler use rather than for use in application code.
Если у вас с английским нормально, советую это видео посмотреть: https://channel9.msdn.com/Even... /TOOL-829T

Сильно крутого английского не надо, достаточно улавливать знакомые слова и смотреть на код.
Еще есть другая классная статья, она немного не про async/await, а скорее отличие асинхронности от многопоточности, но она мне ОЧЕНЬ понравилась, я до этого подобной информации не встречал. Вот она.
2
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
BasicMan
Эксперт
29316 / 5623 / 2384
Регистрация: 17.02.2009
Сообщений: 30,364
Блог
03.06.2016, 20:21
Помогаю со студенческими работами здесь

Как выполняется метод, использующий async и await?
НаMSDN есть такой пример: async Task&lt;int&gt; AccessTheWebAsync() { HttpClient client = new HttpClient(); Task&lt;string&gt;...

Async/await. Как сделать метод асинхронным?
Есть метод, который получает html private static string GetHtml(string id) { using (var client = new...

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

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

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


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

Или воспользуйтесь поиском по форуму:
8
Ответ Создать тему
Новые блоги и статьи
Подключение Box2D v3 к SDL3 для Android: физика и отрисовка коллайдеров
8Observer8 29.01.2026
Содержание блога Box2D - это библиотека для 2D физики для анимаций и игр. С её помощью можно определять были ли коллизии между конкретными объектами. Версия v3 была полностью переписана на Си, в. . .
Инструменты COM: Сохранение данный из VARIANT в файл и загрузка из файла в VARIANT
bedvit 28.01.2026
Сохранение базовых типов COM и массивов (одномерных или двухмерных) любой вложенности (деревья) в файл, с возможностью выбора алгоритмов сжатия и шифрования. Часть библиотеки BedvitCOM Использованы. . .
Загрузка PNG с альфа-каналом на SDL3 для Android: с помощью SDL_LoadPNG (без SDL3_image)
8Observer8 28.01.2026
Содержание блога SDL3 имеет собственные средства для загрузки и отображения PNG-файлов с альфа-каналом и базовой работы с ними. В этой инструкции используется функция SDL_LoadPNG(), которая. . .
Загрузка PNG с альфа-каналом на SDL3 для Android: с помощью SDL3_image
8Observer8 27.01.2026
Содержание блога SDL3_image - это библиотека для загрузки и работы с изображениями. Эта пошаговая инструкция покажет, как загрузить и вывести на экран смартфона картинку с альфа-каналом, то есть с. . .
Влияние грибов на сукцессию
anaschu 26.01.2026
Бифуркационные изменения массы гриба происходят тогда, когда мы уменьшаем массу компоста в 10 раз, а скорость прироста биомассы уменьшаем в три раза. Скорость прироста биомассы может уменьшаться за. . .
Воспроизведение звукового файла с помощью SDL3_mixer при касании экрана Android
8Observer8 26.01.2026
Содержание блога SDL3_mixer - это библиотека я для воспроизведения аудио. В отличие от инструкции по добавлению текста код по проигрыванию звука уже содержится в шаблоне примера. Нужно только. . .
Установка Android SDK, NDK, JDK, CMake и т.д.
8Observer8 25.01.2026
Содержание блога Перейдите по ссылке: https:/ / developer. android. com/ studio и в самом низу страницы кликните по архиву "commandlinetools-win-xxxxxx_latest. zip" Извлеките архив и вы увидите. . .
Вывод текста со шрифтом TTF на Android с помощью библиотеки SDL3_ttf
8Observer8 25.01.2026
Содержание блога Если у вас не установлены Android SDK, NDK, JDK, и т. д. то сделайте это по следующей инструкции: Установка Android SDK, NDK, JDK, CMake и т. д. Сборка примера Скачайте. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru