Synchronous or asynchronous continuation upon TaskCompletionSource.TrySetResult?

后端 未结 1 782
抹茶落季
抹茶落季 2021-01-06 06:49

How to tell if the continuation initiated by TaskCompletionSource.TrySetResult is going to be executed synchronously or asynchronously?

For example:



        
1条回答
  •  栀梦
    栀梦 (楼主)
    2021-01-06 07:04

    I don't think there is a documented way to predict synchronous/asynchronous behaviour of SetResult in advance. If you want to explicitly impose asynchronous continuation, the Task.Run(() => tcs.SetResult()) idea proposed by @Damien_The_Unbeliever is simple and universal.

    However, if you really would like to reduce thread switching and still force asynchrony, you could wrap tcs.SetResult with a custom dumb SynchronizationContext. The only purpose of it would be its uniqueness as compared to the context of await tcs.Task (and to that of any other continuations possibly registered on tcs.Task). That would cause asynchronous continuation(s) on the consumer side of TaskCompletionSource, either via SynchronizationContext.Post or on a pool thread (if there's no synchronization context on the consumer side).

    The test app:

    using System;
    using System.Diagnostics;
    using System.Threading;
    using System.Threading.Tasks;
    using System.Windows.Forms;
    
    namespace WinForms_21845495
    {
        public partial class MainForm : Form
        {
            public MainForm()
            {
                InitializeComponent();
    
                this.Load += async (s, e) =>
                {
                    // test on WindowsFormsSynchronizationContext
                    await RequestNotificationAsync(notifyAsync: false);
                    Debug.WriteLine(String.Empty);
    
                    await RequestNotificationAsync(notifyAsync: true);
                    Debug.WriteLine(String.Empty);
    
                    // test on a pool thread
                    await Task.Run(() => RequestNotificationAsync(notifyAsync: false));
                    Debug.WriteLine(String.Empty);
    
                    await Task.Run(() => RequestNotificationAsync(notifyAsync: true));
                    Debug.WriteLine(String.Empty);
                };
            }
    
            async Task RegisterNotification(TaskCompletionSource tcs, bool notifyAsync)
            {
                await Task.Delay(500);
    
                Debug.WriteLine("A.before");
    
                if (notifyAsync)
                {
                    tcs.SetResultAsync(null);
                }
                else
                {
                    tcs.SetResult(null);
                }
    
                Debug.WriteLine("A.after");
            }
    
            async Task RequestNotificationAsync(bool notifyAsync)
            {
                var tcs = new TaskCompletionSource();
                var task = this.RegisterNotification(tcs, notifyAsync);
    
                Debug.WriteLine("B.before");
    
                var data = await tcs.Task;
    
                // do not yeild
                Thread.Sleep(500); 
                Debug.WriteLine("B.after");
    
                // yeild
                await Task.Delay(500); 
            }
        }
    
        public static class TaskExt
        {
            static public void SetResultAsync(this TaskCompletionSource tcs, T result)
            {
                FakeSynchronizationContext.Execute(() => tcs.SetResult(result));
            }
    
            // FakeSynchronizationContext
            class FakeSynchronizationContext : SynchronizationContext
            {
                private static readonly ThreadLocal s_context =
                    new ThreadLocal(() => new FakeSynchronizationContext());
    
                private FakeSynchronizationContext() { }
    
                public static FakeSynchronizationContext Instance { get { return s_context.Value; } }
    
                public static void Execute(Action action)
                {
                    var savedContext = SynchronizationContext.Current;
                    SynchronizationContext.SetSynchronizationContext(FakeSynchronizationContext.Instance);
                    try
                    {
                        action();
                    }
                    finally
                    {
                        SynchronizationContext.SetSynchronizationContext(savedContext);
                    }
                }
    
                // SynchronizationContext methods
    
                public override SynchronizationContext CreateCopy()
                {
                    return this;
                }
    
                public override void OperationStarted()
                {
                    throw new NotImplementedException("OperationStarted");
                }
    
                public override void OperationCompleted()
                {
                    throw new NotImplementedException("OperationCompleted");
                }
    
                public override void Post(SendOrPostCallback d, object state)
                {
                    throw new NotImplementedException("Post");
                }
    
                public override void Send(SendOrPostCallback d, object state)
                {
                    throw new NotImplementedException("Send");
                }
            }
        }
    }
    
    
    

    The output:

    B.before
    A.before
    B.after
    A.after
    
    B.before
    A.before
    A.after
    B.after
    
    B.before
    A.before
    B.after
    A.after
    
    B.before
    A.before
    A.after
    B.after
    

    0 讨论(0)
    提交回复
    热议问题