System.Reactive 6.0.1
Rx (Reactive Extensions for .NET)
Rx enables event-driven programming with a composable, declarative model.
Getting started
Run the following at a command line:
mkdir TryRx
cd TryRx
dotnet new console
dotnet add package System.Reactive
Alternatively, if you have Visual Studio installed, create a new .NET Console project, and then use the NuGet package manager to add a reference to System.Reactive
.
You can then add this code to your Program.cs
. This creates an observable source (ticks
) that produces an event once every second. It also adds a handler to that source that writes a message to the console for each event:
using System.Reactive.Linq;
IObservable<long> ticks = Observable.Timer(
dueTime: TimeSpan.Zero,
period: TimeSpan.FromSeconds(1));
ticks.Subscribe(
tick => Console.WriteLine($"Tick {tick}"));
Console.ReadLine();
Examples
Wrapping an existing event source as an Rx IObservable<T>
If you have an existing source of events that does not support Rx directly, but which does offer .NET events, you can bring this into the world of Rx using the Observable.FromEventPattern
method:
using System.Reactive.Linq;
FileSystemWatcher fsw = new FileSystemWatcher(@"C:\temp");
IObservable<FileSystemEventArgs> changeEvents = Observable
.FromEventPattern<FileSystemEventHandler, FileSystemEventArgs>(
h => fsw.Changed += h,
h => fsw.Changed -= h)
.Select(e => e.EventArgs);
fsw.EnableRaisingEvents = true;
Waiting for inactivity
It can sometimes be useful to wait for a period of inactivity before taking action. For example, if you have code that monitors a directory in a filesystem, processing modified or added files, it's common for there to be flurries of activity. For example, if a user is copying multiple files into a folder that you're observing, there will be multiple changes, and it could be more efficient to wait until those stop and then process all the changes in one batch, than to attempt to process everything immediately.
This example defines a custom Rx operator that can be attached to any source. It will wait for that source to start producing events, and then, it will wait for it to stop again for the specified period. Each time that happens, it reports all of the activity that occurred between the last two periods of inactivity:
static class RxExt
{
public static IObservable<IList<T>> Quiescent<T>(
this IObservable<T> src,
TimeSpan minimumInactivityPeriod,
IScheduler scheduler)
{
IObservable<int> onoffs =
from _ in src
from delta in Observable.Return(1, scheduler).Concat(Observable.Return(-1, scheduler).Delay(minimumInactivityPeriod, scheduler))
select delta;
IObservable<int> outstanding = onoffs.Scan(0, (total, delta) => total + delta);
IObservable<int> zeroCrossings = outstanding.Where(total => total == 0);
return src.Buffer(zeroCrossings);
}
}
(This works by creating a sequence (onoffs
) that produces a value 1 each time activity occurs, and then a corresponding -1 after the specified time has elapsed. It then uses Scan
to produce the outstanding
sequence, which is just a running total of those onoffs
. This is effectively a count of the number of events that have happened recently (where 'recently' is defined as 'less than minimumInactivityPeriod
ago). Every new event that occurs raises this running total by 1, but each time the specified timespan has passed for a particular event, it drops by one. So when this drops back to 0, it means that there are no events that have occurred as recently as the minimumInactivityPeriod
. The zeroCrossings
sequence picks out just the events in which outstanding
drops back to zero. This has the effect that zeroCrossings
raises an event every time there has been some activity followed by minimumInactivityPeriod
of inactivity. Finally, we plug this into the Buffer
operator, which slices the input events (src
) into chunks. By passing it the zeroCrossings
source, we tell Buffer
to deliver a new slice every time the source becomes inactive. The effect is that the source returned by Quiescent
does nothing until there has been some activity followed by the specified period of inactivity, at which point it produces a single event reporting all of the source events that have occurred since the previous period, or in the initial case, all of the source events so far.)
You could use this in conjunction with the adapted FileSystemWatcher
from the preceding example:
IObservable<IList<FileSystemEventArgs>> fileActivityStopped = changeEvents
.Quiescent(TimeSpan.FromSeconds(2), Scheduler.Default);
await fileActivityStopped.ForEachAsync(
a => Console.WriteLine($"File changes stopped after {a.Count} changes"));
(Note: this only uses the Changed
event. A real application might also need to look at the FileSystemWatcher
's Created
, Renamed
, and Deleted
events.)
Feedback
You can create issues at the https://github.com/dotnet/reactive repository
Showing the top 20 packages that depend on System.Reactive.
Packages | Downloads |
---|---|
NBomber
Modern and flexible load testing framework for Pull and Push scenarios, designed to test any system regardless a protocol (HTTP/WebSockets/AMQP etc) or a semantic model (Pull/Push).
|
3 |
ExtentReports
ExtentReports .Net Standard
|
3 |
System.Reactive.Interfaces
Legacy facade for Reactive Extensions (Rx) for .NET
|
3 |
FSharp.Control.Reactive
A F#-friendly wrapper for the Reactive Extensions.
|
3 |
System.Reactive.Core
Legacy facade for Reactive Extensions (Rx) for .NET
|
3 |
System.Reactive.WindowsRuntime
Reactive Extensions (Rx) for .NET - v3 compatibility facade for
|
3 |
System.Reactive.Linq
Reactive Extensions (Rx) for .NET - v3 compatibility facade for
|
3 |
System.Reactive.Core
Reactive Extensions (Rx) for .NET - v3 compatibility facade for
|
3 |
System.Reactive.Interfaces
Reactive Extensions (Rx) for .NET - v3 compatibility facade for
|
3 |
.NET Framework 4.7.2
- System.Threading.Tasks.Extensions (>= 4.5.4)
.NET 6.0
- No dependencies.
.NET 6.0
- No dependencies.
.NET Standard 2.0
- System.Threading.Tasks.Extensions (>= 4.5.4)
UAP 10.0.18362
- System.Threading.Tasks.Extensions (>= 4.5.4)
Version | Downloads | Last updated |
---|---|---|
6.0.1 | 4 | 10/05/2024 |
6.0.1-preview.1 | 4 | 10/06/2024 |
6.0.0 | 4 | 03/27/2024 |
6.0.0-preview.16 | 2 | 11/19/2024 |
6.0.0-preview.13 | 2 | 11/08/2024 |
6.0.0-preview.9 | 2 | 11/17/2024 |
6.0.0-preview.1 | 3 | 10/06/2024 |
5.0.0 | 2 | 11/25/2024 |
5.0.0-preview.220 | 2 | 11/18/2024 |
5.0.0-preview.16 | 2 | 11/19/2024 |
4.4.1 | 3 | 03/06/2023 |
4.3.2 | 2 | 11/25/2024 |
4.3.1 | 2 | 10/12/2024 |
4.2.2 | 2 | 11/18/2024 |
4.2.0 | 2 | 11/25/2024 |
4.2.0-preview.625 | 3 | 10/06/2024 |
4.2.0-preview.566 | 3 | 10/06/2024 |
4.2.0-preview.102 | 2 | 11/18/2024 |
4.2.0-preview.63 | 2 | 11/19/2024 |
4.1.6 | 3 | 10/06/2024 |
4.1.5 | 3 | 10/06/2024 |
4.1.4 | 2 | 11/25/2024 |
4.1.3 | 3 | 10/06/2024 |
4.1.2 | 3 | 10/06/2024 |
4.1.1 | 2 | 11/25/2024 |
4.1.0 | 2 | 11/25/2024 |
4.1.0-preview.330 | 1 | 11/18/2024 |
4.1.0-preview.84 | 2 | 11/19/2024 |
4.0.0 | 2 | 11/25/2024 |
4.0.0-preview.4.build.391 | 2 | 10/06/2024 |
4.0.0-preview.3.build.380 | 2 | 10/06/2024 |
4.0.0-preview.2.build.379 | 2 | 10/06/2024 |
3.1.1 | 2 | 11/25/2024 |
3.1.0 | 2 | 11/25/2024 |
3.1.0-rc | 2 | 11/25/2024 |
3.0.0 | 3 | 10/06/2024 |