Gapotchenko.FX.Threading
2024.2.5
Prefix Reserved
dotnet add package Gapotchenko.FX.Threading --version 2024.2.5
NuGet\Install-Package Gapotchenko.FX.Threading -Version 2024.2.5
<PackageReference Include="Gapotchenko.FX.Threading" Version="2024.2.5" />
paket add Gapotchenko.FX.Threading --version 2024.2.5
#r "nuget: Gapotchenko.FX.Threading, 2024.2.5"
// Install Gapotchenko.FX.Threading as a Cake Addin #addin nuget:?package=Gapotchenko.FX.Threading&version=2024.2.5 // Install Gapotchenko.FX.Threading as a Cake Tool #tool nuget:?package=Gapotchenko.FX.Threading&version=2024.2.5
Overview
The module provides complementary primitives for multithreaded and asynchronous programming in .NET.
TaskBridge
TaskBridge
class from Gapotchenko.FX.Threading
module provides seamless interoperability between synchronous and asynchronous code execution models.
Executing an async task from synchronous code poses a few rather big challenges in conventional .NET:
- The wait operation for an async task is prone to deadlocks unless a proper synchronization context is in place
- The cancellation models of sync and async code are different and often incompatible
Meet TaskBridge
. It makes interoperability a breeze:
using Gapotchenko.FX.Threading.Tasks;
using System;
using System.Threading.Tasks;
class Program
{
static void Main()
{
TaskBridge.Execute(RunAsync);
}
static async Task RunAsync()
{
await Console.Out.WriteLineAsync("Hello, Async World!");
}
}
<details> <summary>More details on TaskBridge</summary>
Cancellation Models
TaskBridge
provides automatic interoperability between different cancellation models.
Let's call a cancelable async method from a synchronous thread that can be aborted by Thread.Abort()
method:
using Gapotchenko.FX.Threading.Tasks;
using System.Threading;
using System.Threading.Tasks;
void SyncMethod() // can be canceled by Thread.Abort()
{
// Executes an async task that is gracefully canceled via cancellation
// token when current thread is being aborted or interrupted.
TaskBridge.Execute(DoJobAsync); // <-- TaskBridge DOES THE MAGIC
}
async Task DoJobAsync(CancellationToken ct)
{
…
// Gracefully handles cancellation opportunities.
ct.ThrowIfCancellationRequested();
…
}
You see this? A simple one-liner for a complete interoperability between two execution models.
Now, let's take a look at the opposite scenario where a cancelable async task calls an abortable synchronous code:
using Gapotchenko.FX.Threading.Tasks;
using System.Threading;
using System.Threading.Tasks;
async Task DoJobAsync(CancellationToken ct) // can be canceled by a specified cancellation token
{
// Executes a synchronous method that is thread-aborted when
// a specified cancellation token is being canceled.
await TaskBridge.ExecuteAsync(SyncMethod, ct); // <-- TaskBridge DOES THE MAGIC
}
void SyncMethod()
{
…
}
As you can see, TaskBridge
has a lot of chances to become your tool #1,
as it elegantly solves a world-class problem of bridging sync and async models together.
</details>
Sequential
, an Antogonist to Parallel
.NET platform provides System.Threading.Tasks.Parallel
class that contains a bunch of static methods allowing one to execute the tasks in parallel.
But what if you want to temporarily switch them to a sequential execution mode?
Of course, you can do that manually, for example, by changing Parallel.ForEach
method to foreach
C# language keyword.
But this constitutes a lot of manual labour prone to errors.
That's why Gapotchenko.FX.Threading
module provides Sequential
class, a drop-in anotogonist to Parallel
.
It allows you to make the switch by just changing the class name from Parallel
to Sequential
in a corresponding function call.
So Parallel.ForEach
becomes Sequential.ForEach
, and voila, the tasks are now executed sequentially allowing you to isolate that pesky multithreading bug you were hunting for.
Automatic Selection Between Parallel and Sequential
System.Threading.Tasks.Parallel
class allows you to execute tasks in parallel,
while Gapotchenko.FX.Threading.Tasks.Sequential
allows you to do the same,
but sequentially.
But what if you want to get the best of two worlds?
Meet DebuggableParallel
class provided by Gapotchenko.FX.Threading
module that does an automatic contextful choice for you.
<details> <summary>More details</summary>
When a project has an attached debugger, DebuggableParallel
primitive executes the specified tasks sequentially.
When there is no debugger attached, DebuggableParallel
will execute the tasks in parallel.
And of course, it's a drop-in replacement for the ubiquitous System.Threading.Tasks.Parallel
class.
The automatic selection of the task execution mode enables multi-threaded code to be effortlessly debugged,
while preserving the multi-core efficiency when no debugger is present.
The selection can also be overridden from code.
For example, if you want to disallow that debugger friendliness in Release
configuration,
you can correspondingly configure the DebuggableParallel
class at the very start of a program:
using Gapotchenko.FX.Threading.Tasks;
#if !DEBUG
DebuggableParallel.Mode = DebuggableParallelMode.AlwaysParallel;
#endif
This makes the behavior of DebuggableParallel
class to be essentially indistinguishable from System.Threading.Tasks.Parallel
without changing any other code.
</details>
Asynchronous Concurrency
Gapotchenko.FX.Threading
module provides plenty of synchronization primitives supporting not only synchronous, but also asynchronous execution models.
This closes the gap in the mainstream .NET BCL which had a decade-old lack of them.
<details> <summary>Historical context</summary>
One of the main barriers for implementing asynchronous synchronization in .NET was the impossibility to achieve reentrancy.
That impossibility was caused by certain limitations of System.AsyncLocal<T>
class that only supported downward propagation of information associated with an asynchronous control flow.
However, using the tradition of rigorous and meticulous mathematical problem solving, Gapotchenko.FX.Threading
module became the world's first (clean) implementation of reentrant synchronization primitives for .NET's asynchronous execution model.
The word "clean" means that it does not use such unreliable techniques as System.Diagnostics.StackTrace
.
Previously, clean implementations were considered impossible due to aforementioned limitations of the System.AsyncLocal<T>
class.
If you are interested in gory implementation details, take a look at the corresponding source file.
</details>
AsyncLock
Gapotchenko.FX.Threading.AsyncLock
represents a reentrant synchronization primitive
that ensures that only one task or thread can access a resource at any given time:
using Gapotchenko.FX.Threading;
var lockObj = new AsyncLock();
await lockObj.EnterAsync();
try
{
// Only one task can execute this section of code at any given time.
// ...
}
finally
{
lockObj.Exit();
}
AsyncLock
implements Gapotchenko.FX.Threading.IAsyncLockable
interface that gives you access to handy shortcuts for entering and exiting the lock scope asynchronously without manual try
/finally
constructs:
using Gapotchenko.FX.Threading;
var lockObj = new AsyncLock();
using (await lockObj.EnterScopeAsync())
{
// Only one task can execute this section of code at any given time.
// ...
}
The AsyncLock
can be acquired multiple times by the same task because the primitive supports reentrancy (which is also called recursion):
using Gapotchenko.FX.Threading;
var lockObj = new AsyncLock();
using (await lockObj.EnterScopeAsync())
{
using (await lockObj.EnterScopeAsync())
{
// Only one task can execute this section of code at any given time.
// ...
}
}
Gapotchenko.FX.Threading.AsyncLock
is an asynchronous equivalent of System.Threading.Lock
class.
AsyncCriticalSection
Gapotchenko.FX.Threading.AsyncCriticalSection
represents a non-reentrant synchronization primitive
that ensures that only one task or thread can access a resource at any given time.
AsyncCriticalSection
is a non-reentrant variant of the AsyncLock
class:
using Gapotchenko.FX.Threading;
var cs = new AsyncCriticalSection();
using (await cs.EnterScopeAsync())
{
// Only one task can execute this section of code at any given time.
// ...
}
The benefit of using AsyncCriticalSection
in comparison to AsyncLock
is that the former uses fewer computational resources because it has no need to track reentrancy.
So if you know that your algorithm does not require recursive locking, using AsyncCriticalSection
is more preferable.
AsyncManualResetEvent
Gapotchenko.FX.Threading.AsyncManualResetEvent
represents a synchronization primitive that, when signaled, allows one or more tasks waiting on it to proceed.
AsyncManualResetEvent
must be reset manually.
This is an asynchronous equivalent of System.Threading.ManualResetEvent
class.
AsyncAutoResetEvent
Gapotchenko.FX.Threading.AsyncAutoResetEvent
represents a synchronization primitive that, when signaled, allows one or more tasks waiting on it to proceed.
AsyncAutoResetEvent
resets automatically after releasing a single waiting task.
This is an asynchronous equivalent of System.Threading.AutoResetEvent
class.
AsyncMonitor
Gapotchenko.FX.Threading.AsyncMonitor
represents a reentrant concurrency primitive that provides a mechanism that synchronizes access to objects:
using Gapotchenko.FX.Threading;
// Only one task can access this variable at any given time.
int? result = null;
var monitor = new AsyncMonitor();
// Start consuming and producing tasks in parallel.
await Task.WhenAll(
Consume("Consumer #1"),
Consume("Consumer #2"),
Produce());
async Task Consume(string taskName)
{
using (await monitor.EnterScopeAsync())
{
while (result == null)
await monitor.WaitAsync();
await Console.Out.WriteLineAsync($"Result for {taskName}: {result}");
}
}
async Task Produce()
{
await Task.Delay(1000); // pretend that it takes time to come up with a result
using (await monitor.EnterScopeAsync())
{
result = 42;
monitor.NotifyAll();
}
}
The example code above produces the following output after one second of waiting:
Result for Consumer #1: 42
Result for Consumer #2: 42
Gapotchenko.FX.Threading.AsyncMonitor
is an asynchronous equivalent of System.Threading.Monitor
class.
<details> <summary>More details on AsyncMonitor</summary>
Gapotchenko.FX.Threading.AsyncMonitor
class can be used as a drop-in replacement for System.Threading.Monitor
.
Let's take a look at example.
The synchronous code below uses System.Threading.Monitor
:
class OldCode
{
Queue<int> m_WorkItems = new();
public void AddWorkItem(int item)
{
lock (m_WorkItems)
{
m_WorkItems.Enqueue(item);
Monitor.Pulse(m_WorkItems);
}
}
public int GetWorkItem()
{
lock (m_WorkItems)
{
while (m_WorkItems.Count == 0) // wait for items to appear in the queue
Monitor.Wait(m_WorkItems);
return m_WorkItems.Dequeue();
}
}
}
It can be translated to asynchronous code using Gapotchenko.FX.Threading.AsyncMonitor
class, almost word for word:
class NewCode
{
Queue<int> m_WorkItems = new();
public async Task AddWorkItemAsync(int item)
{
var monitor = AsyncMonitor.For(m_WorkItems);
using (await monitor.EnterScopeAsync())
{
m_WorkItems.Enqueue(item);
monitor.Notify();
}
}
public async Task<int> GetWorkItemAsync()
{
var monitor = AsyncMonitor.For(m_WorkItems);
using (await monitor.EnterScopeAsync())
{
while (m_WorkItems.Count == 0) // wait for items to appear in the queue
await monitor.WaitAsync();
return m_WorkItems.Dequeue();
}
}
// Note that it's possible to mix asynchronous and synchronous execution models
// when using synchronization primitives provided by Gapotchenko.FX.Threading module.
// Both models happily coexist in terms of concurrency.
//
// For example, that useful possibility may be used for a gradual codebase migration
// from synchronous execution model to asynchronous.
public void AddWorkItem(int item)
{
var monitor = AsyncMonitor.For(m_WorkItems);
using (monitor.EnterScope())
{
m_WorkItems.Enqueue(item);
monitor.Notify();
}
}
public int GetWorkItem()
{
var monitor = AsyncMonitor.For(m_WorkItems);
using (monitor.EnterScope())
{
while (m_WorkItems.Count == 0) // wait for items to appear in the queue
monitor.Wait();
return m_WorkItems.Dequeue();
}
}
}
Note the use of AsyncMonitor.For(object)
method in the code above.
It allows to imitate the semantics associated with lock (object)
C# keyword that attaches a monitor to the specified object.
AsyncMonitor.For(object)
method is provided for semantical and ideological compatibility with System.Threading.Monitor
class in order to ease the translation of existing codebases.
As a more efficient approach, however, it is recommended to use an instance of AsyncMonitor
explicitly without "attaching" it to a particular object:
class NewCode
{
Queue<int> m_WorkItems = new();
AsyncMonitor m_Monitor = new();
public async Task AddWorkItemAsync(int item)
{
using (await m_Monitor.EnterScopeAsync())
{
m_WorkItems.Enqueue(item);
m_Monitor.Notify();
}
}
// ...
}
</details>
Commonly Used Types
Gapotchenko.FX.Threading.TaskBridge
Gapotchenko.FX.Threading.AsyncLock
Gapotchenko.FX.Threading.AsyncMonitor
Other Modules
Let's continue with a look at some other modules provided by Gapotchenko.FX:
- Gapotchenko.FX
- Gapotchenko.FX.AppModel.Information
- Gapotchenko.FX.Collections
- Gapotchenko.FX.Console
- Gapotchenko.FX.Data
- Gapotchenko.FX.Diagnostics
- Gapotchenko.FX.IO
- Gapotchenko.FX.Linq
- Gapotchenko.FX.Math
- Gapotchenko.FX.Memory
- Gapotchenko.FX.Security.Cryptography
- Gapotchenko.FX.Text
- ➴ Gapotchenko.FX.Threading
- Gapotchenko.FX.Tuples
Or look at the full list of modules.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 is compatible. net5.0-windows was computed. net6.0 is compatible. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 is compatible. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 is compatible. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 is compatible. netcoreapp2.2 was computed. netcoreapp3.0 is compatible. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 is compatible. |
.NET Framework | net461 is compatible. net462 was computed. net463 was computed. net47 was computed. net471 is compatible. net472 is compatible. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETCoreApp 2.1
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
.NETCoreApp 3.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
.NETFramework 4.6.1
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
.NETFramework 4.7.1
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
.NETFramework 4.7.2
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
.NETStandard 2.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
.NETStandard 2.1
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
net5.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
net6.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
net7.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
net8.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
-
net9.0
- Gapotchenko.FX (>= 2024.2.5)
- Gapotchenko.FX.Collections (>= 2024.2.5)
NuGet packages (3)
Showing the top 3 NuGet packages that depend on Gapotchenko.FX.Threading:
Package | Downloads |
---|---|
Gapotchenko.FX.Diagnostics.Process
Provides extended functionality for process manipulation. |
|
Gapotchenko.FX.Profiles.Core
Represents the Core profile of Gapotchenko.FX. |
|
Gapotchenko.FX.Data.Linq
Provides asynchronous data access support for LINQ2SQL technology. |
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
2024.2.5 | 254 | 12/31/2024 |
2024.1.3 | 1,339 | 11/10/2024 |
2022.2.7 | 42,111 | 5/1/2022 |
2022.2.5 | 2,740 | 5/1/2022 |
2022.1.4 | 2,878 | 4/6/2022 |
2021.2.21 | 3,024 | 1/21/2022 |
2021.2.20 | 2,847 | 1/17/2022 |
2021.1.5 | 8,721 | 7/6/2021 |
2020.2.2-beta | 1,125 | 11/21/2020 |
2020.1.15 | 5,312 | 11/5/2020 |
2020.1.9-beta | 1,208 | 7/14/2020 |
2020.1.8-beta | 1,202 | 7/14/2020 |
2020.1.7-beta | 1,242 | 7/14/2020 |
2020.1.1-beta | 1,189 | 2/11/2020 |
2019.3.7 | 2,374 | 11/4/2019 |
2019.2.20 | 1,533 | 8/13/2019 |
2019.1.151 | 26,774 | 3/30/2019 |