Когда я буду использовать Task.Yield()?

Я использую async/wait и Task много, но никогда не использовал Task.Yield() и, честно говоря, даже со всеми объяснениями, которые я не понимаю, зачем мне нужен этот метод.

Может ли кто-нибудь дать хороший пример, где Yield() требуется?

Ответ 1

Когда вы используете async/await, нет гарантии, что метод, который вы вызываете, когда вы выполняете await FooAsync(), будет выполняться асинхронно. Внутренняя реализация может быть возвращена с использованием полностью синхронного пути.

Если вы создаете API, где это критически важно, что вы не блокируете, и вы запускаете некоторый код асинхронно, и есть вероятность, что вызываемый метод будет работать синхронно (эффективно блокирует), используя await Task.Yield(), заставит ваш метод быть асинхронным и возвращать управление в этой точке. Остальная часть кода будет выполнена позже (в этот момент она все равно может работать синхронно) в текущем контексте.

Это также может быть полезно, если вы создаете асинхронный метод, который требует некоторой "длительной" инициализации, то есть:

 private async void button_Click(object sender, EventArgs e)
 {
      await Task.Yield(); // Make us async right away

      var data = ExecuteFooOnUIThread(); // This will run on the UI thread at some point later

      await UseDataAsync(data);
 }

Без вызова Task.Yield() метод будет выполняться синхронно вплоть до первого вызова await.

Ответ 2

Внутренне, await Task.Yield() просто ставит в очередь продолжение либо в текущем контексте синхронизации, либо в потоке случайного пула, если SynchronizationContext.Current равен null.

Он эффективно реализован как пользовательский ожидающий. Менее эффективный код, производящий идентичный эффект, может быть таким простым:

var tcs = new TaskCompletionSource<bool>();
var sc = SynchronizationContext.Current;
if (sc != null)
    sc.Post(_ => tcs.SetResult(true), null);
else
    ThreadPool.QueueUserWorkItem(_ => tcs.SetResult(true));
await tcs.Task;

Task.Yield() можно использовать как ярлык для некоторых странных изменений потока выполнения. Например:

async Task DoDialogAsync()
{
    var dialog = new Form();

    Func<Task> showAsync = async () => 
    {
        await Task.Yield();
        dialog.ShowDialog();
    }

    var dialogTask = showAsync();
    await Task.Yield();

    // now we're on the dialog nested message loop started by dialog.ShowDialog 
    MessageBox.Show("The dialog is visible, click OK to close");
    dialog.Close();

    await dialogTask;
    // we're back to the main message loop  
}

Тем не менее, я не могу вспомнить ни одного случая, когда Task.Yield() нельзя заменить на Task.Factory.StartNew с правильным планировщиком задач.

Смотрите также:

Ответ 3

Одним из применений Task.Yield() является предотвращение при выполнении асинхронной рекурсии. Task.Yield() предотвращает синхронное продолжение.

private static void Main()
    {
        RecursiveMethod().Wait();
    }

    private static async Task RecursiveMethod()
    {
        await Task.Delay(1);
        //await Task.Yield(); // Uncomment this line to prevent stackoverlfow.
        await RecursiveMethod();
    }

Ответ 4

У меня есть базовый класс, на котором есть первичный метод:

public async Task SendAsync (message); 

Затем я решил позже указать, что мне нужно было проверить сообщение, отправленное в более продвинутом варианте базового класса, к которому я добавил два базовых метода:

protected virtual async Task OnBeforeSend(message);
protected virtual async Task OnAfterSend(message);

В специализированной версии моего класса я добавил реализации методов до и после отправки сообщения, но не нужно было добавлять пользовательские методы до и после отправки в мою базовую реализацию класса. Мне потребовалось, чтобы моя базовая реализация также была конкретного типа, поскольку она использовалась где-то еще.

Чтобы сохранить компилятор счастливым, мне нужно было предоставить результат задачи для базовых (пустых) реализаций, и для этого я использовал подход Task.Yield(). Поэтому в моей реализации базового класса у меня есть следующее:

protected virtual async Task OnBeforeSend(IEmailMessage mailMessage)
{
    await Task.Yield();
}

Кажется, это хорошо работает для меня, и я не вижу причины, почему это было бы неправильно.

Ответ 5

Task.Yield() может использоваться в имитационных реализациях асинхронных методов.