Pavel, My understanding might be wrong but I think the best practice (or even strongly recommended way) to implement async methods in .NET is to execute continuation on the caller's thread if ConfigureAwait(false) was not specified. Pseudo-code might look like:
async Task PutAsync(K k, V v) { var continuationExecutor = configureAwait ? (SynchronizationContext.Current ?? TaskScheduler.Current) : null; await <<async implementation>> continuationExecutor.Post(continuation); } I got this understanding from reading some blog about SynchronizationContext lots of time ago. They were saying they created SynchronizationContext specifically to allow posting continuations to the caller's thread. The reason for that is to simplify the user's code to avoid routing in the code. Suppose you have a UI (like WPF or WinForms) event handler that must be processed on the U thread: async Task Button1_Click(EventArgs args) { ignite.PutAsync(args.Key, args.Value); Button1.Disabled = true; } Executing the "Button1.Disabled = true" on a ForkJoinPool pool would cause a "Trying to modify UI on a non-UI thread" exception. But if you capture SynchronizationContext.Current in PutAsync and then route continuation to the captured context then the code would work. I think the users really expect the continuations to be executed on the caller's thread. Sometimes you know that your continuation is really fast and safe and you want to avoid switching threads to improve performance. In this case you use ConfigureAwait(false) like ignite.PutAsync(args.Key, args.Value).ConfigureAwat(false); In this case the continuation executes on the Ignite thread without routing to the caller's thread. вт, 16 мар. 2021 г. в 18:49, Pavel Tupitsyn <ptupit...@apache.org>: > Alexey, > > .NET thick API delegates to Java directly. > > When you do ICache.PutAsync(): > * Future is created on Java side, .listen() is called > * TaskCompletionSource is created on .NET side, its Task is returned to the > user > * Operation completes, Future listener is called on the Java side > * Listener invokes JNI callback to .NET, where > TaskCompletionSource.SetResult is called > > Therefore, .NET user code (in ContinueWith or after await) will be executed > on the Java > thread that invokes the future listener. > > After the proposed fix, future listeners will be invoked on > ForkJoinPool#commonPool (instead of striped pool). > So .NET continuations will end up in commonPool as well, which solves the > problem for .NET automatically, no changes required. > > Does that make sense? > > On Tue, Mar 16, 2021 at 1:52 PM Alexey Kukushkin < > kukushkinale...@gmail.com> > wrote: > > > Hi Pavel, > > > > Extending Java async API with additional Executor parameters looks OK to > > me. > > > > It is not clear from the IEP how you are going to do that for .NET async > > API. My understanding is in .NET we do not add any Executors. Instead, > the > > Ignite Async API should use (SynchronizationContext.Current ?? > > TaskScheduler.Current) by default and it should have exciting behavior > (use > > Ignite striped pool) if ConfigureAwait(false) was specified for the Task > > result. > > > > Is my understanding correct? > > > > > > пн, 15 мар. 2021 г. в 19:24, Pavel Tupitsyn <ptupit...@apache.org>: > > > > > Igniters, > > > > > > Please review the IEP [1] and let me know your thoughts. > > > > > > [1] > > > > > > > > > https://cwiki.apache.org/confluence/display/IGNITE/IEP-70%3A+Async+Continuation+Executor > > > > > > > > > -- > > Best regards, > > Alexey > > > -- Best regards, Alexey