** Minimal memory footprint. Blazing-fast execution. **
Note
If you're curious to see the power of this library, check out the benchmark comparing MediatR vs Mediator Source Generator vs DispatchR.
- Built entirely on top of Dependency Injection
- Zero runtime reflection after registration
- Choose your handler return type:
Task
,ValueTask
, orSynchronous Method
- Allocates nothing on the heap — ideal for high-throughput scenarios
- Outperforms existing solutions in most real-world benchmarks
- Seamlessly compatible with MediatR — migrate with minimal effort
- Currently supports
- Simple Request:
IRequest<TRquest, TResponse>
IRequestHandler<TRequest, TResponse>
IPipelineBehavior<TRequest, TResponse>
- Stream Request:
IStreamRequest<TRquest, TResponse>
IStreamRequestHandler<TRequest, TResponse>
IStreamPipelineBehavior<TRequest, TResponse>
- Notifications:
INotification
INotificationHandler<TRequestEvent>
- Simple Request:
💡 Tip: If you're looking for a mediator with the raw performance of hand-written code, DispatchR is built for you.
In the following, you will see the key differences and implementation details between MediatR and DispatchR.
public sealed class PingMediatR : IRequest<int> { }
- Sending
TRequest
toIRequest
- Precise selection of output for both
async
andsync
handlers- Ability to choose between
Task
andValueTask
- Ability to choose between
public sealed class PingDispatchR : IRequest<PingDispatchR, ValueTask<int>> { }
public sealed class PingHandlerMediatR : IRequestHandler<PingMediatR, int>
{
public Task<int> Handle(PingMediatR request, CancellationToken cancellationToken)
{
return Task.FromResult(0);
}
}
public sealed class PingHandlerDispatchR : IRequestHandler<PingDispatchR, ValueTask<int>>
{
public ValueTask<int> Handle(PingDispatchR request, CancellationToken cancellationToken)
{
return ValueTask.FromResult(0);
}
}
public sealed class LoggingBehaviorMediat : IPipelineBehavior<PingMediatR, int>
{
public Task<int> Handle(PingMediatR request, RequestHandlerDelegate<int> next, CancellationToken cancellationToken)
{
return next(cancellationToken);
}
}
- Use Chain of Responsibility pattern
public sealed class LoggingBehaviorDispatchR : IPipelineBehavior<PingDispatchR, ValueTask<int>>
{
public required IRequestHandler<PingDispatchR, ValueTask<int>> NextPipeline { get; set; }
public ValueTask<int> Handle(PingDispatchR request, CancellationToken cancellationToken)
{
return NextPipeline.Handle(request, cancellationToken);
}
}
- For every kind of return type —
Task
,ValueTask
, or synchronous methods — you need to write a generic pipeline behavior. However, you don't need a separate pipeline for each request. As shown in the code below, this is a GenericPipeline for requests that return aValueTask
.
public class GenericPipelineBehavior<TRequest, TResponse>() : IPipelineBehavior<TRequest, ValueTask<TResponse>>
where TRequest : class, IRequest<TRequest, ValueTask<TResponse>>, new()
{
public required IRequestHandler<TRequest, ValueTask<TResponse>> NextPipeline { get; set; }
public ValueTask<TResponse> Handle(TRequest request, CancellationToken cancellationToken)
{
// You can add custom logic here, like logging or validation
// This pipeline behavior can be used for any request type
return NextPipeline.Handle(request, cancellationToken);
}
}
- DispatchR lets the request itself define the return type.
- No runtime reflection in DispatchR — it's optimized for performance.
- No static behavior chains — pipelines are chained via DI and handler wiring.
- Supports
void
,Task
, orValueTask
as return types.
Ideal for high-performance .NET applications.
public sealed class CounterStreamRequestMediatR : IStreamRequest<int> { }
- Sending
TRequest
toIStreamRequest
public sealed class CounterStreamRequestDispatchR : IStreamRequest<PingDispatchR, ValueTask<int>> { }
public sealed class CounterStreamHandlerMediatR : IStreamRequestHandler<CounterStreamRequestMediatR, int>
{
public async IAsyncEnumerable<int> Handle(CounterStreamRequestMediatR request, CancellationToken cancellationToken)
{
yield return 1;
}
}
public sealed class CounterStreamHandlerDispatchR : IStreamRequestHandler<CounterStreamHandlerDispatchR, int>
{
public async IAsyncEnumerable<int> Handle(CounterStreamHandlerDispatchR request, CancellationToken cancellationToken)
{
yield return 1;
}
}
public sealed class CounterPipelineStreamHandler : IStreamPipelineBehavior<CounterStreamRequestMediatR, string>
{
public async IAsyncEnumerable<string> Handle(CounterStreamRequestMediatR request, StreamHandlerDelegate<string> next, [EnumeratorCancellation] CancellationToken cancellationToken)
{
await foreach (var response in next().WithCancellation(cancellationToken).ConfigureAwait(false))
{
yield return response;
}
}
}
- Use Chain of Responsibility pattern
public sealed class CounterPipelineStreamHandler : IStreamPipelineBehavior<CounterStreamRequestDispatchR, string>
{
public required IStreamRequestHandler<CounterStreamRequestDispatchR, string> NextPipeline { get; set; }
public async IAsyncEnumerable<string> Handle(CounterStreamRequestDispatchR request, [EnumeratorCancellation] CancellationToken cancellationToken)
{
await foreach (var response in NextPipeline.Handle(request, cancellationToken).ConfigureAwait(false))
{
yield return response;
}
}
}
public class GenericStreamPipelineBehavior<TRequest, TResponse>() : IStreamPipelineBehavior<TRequest, TResponse>
where TRequest : class, IStreamRequest<TRequest, TResponse>, new()
{
public IStreamRequestHandler<TRequest, TResponse> NextPipeline { get; set; }
public async IAsyncEnumerable<TResponse> Handle(TRequest request, CancellationToken cancellationToken)
{
await foreach (var response in NextPipeline.Handle(request, cancellationToken).ConfigureAwait(false))
{
yield return response;
}
}
}
public sealed record Event(Guid Id) : INotification;
public sealed class EventHandler(ILogger<Event> logger) : INotificationHandler<Event>
{
public Task Handle(Event notification, CancellationToken cancellationToken)
{
logger.LogInformation("Received notification");
return Task.CompletedTask;
}
}
- Use ValueTask
public sealed record Event(Guid Id) : INotification;
public sealed class EventHandler(ILogger<Event> logger) : INotificationHandler<Event>
{
public ValueTask Handle(Event notification, CancellationToken cancellationToken)
{
logger.LogInformation("Received notification");
return ValueTask.CompletedTask;
}
}
DispatchR is designed with one goal in mind: maximize performance with minimal memory usage. Here's how it accomplishes that:
public TResponse Send<TRequest, TResponse>(IRequest<TRequest, TResponse> request,
CancellationToken cancellationToken) where TRequest : class, IRequest, new()
{
return serviceProvider
.GetRequiredService<IRequestHandler<TRequest, TResponse>>()
.Handle(Unsafe.As<TRequest>(request), cancellationToken);
}
public IAsyncEnumerable<TResponse> CreateStream<TRequest, TResponse>(IStreamRequest<TRequest, TResponse> request,
CancellationToken cancellationToken) where TRequest : class, IStreamRequest, new()
{
return serviceProvider.GetRequiredService<IStreamRequestHandler<TRequest, TResponse>>()
.Handle(Unsafe.As<TRequest>(request), cancellationToken);
}
Only the handler is resolved and directly invoked!
public async ValueTask Publish<TNotification>(TNotification request, CancellationToken cancellationToken) where TNotification : INotification
{
var notificationsInDi = serviceProvider.GetRequiredService<IEnumerable<INotificationHandler<TNotification>>>();
var notifications = Unsafe.As<INotificationHandler<TNotification>[]>(notificationsInDi);
foreach (var notification in notifications)
{
var valueTask = notification.Handle(request, cancellationToken);
if (valueTask.IsCompletedSuccessfully is false) // <-- Handle sync notifications
{
await valueTask;
}
}
}
But the real magic happens behind the scenes when DI resolves the handler dependency:
💡 Tips:
We cache the handler using DI, so in scoped scenarios, the object is constructed only once and reused afterward.
In terms of Dependency Injection (DI), everything in Requests is an IRequestHandler, it's just the keys that differ. When you request a specific key, a set of 1+N objects is returned: the first one is the actual handler, and the rest are the pipeline behaviors.
services.AddScoped(handlerInterface, sp =>
{
var pipelinesWithHandler = Unsafe
.As<IRequestHandler[]>(sp.GetKeyedServices<IRequestHandler>(key));
IRequestHandler lastPipeline = pipelinesWithHandler[0];
for (int i = 1; i < pipelinesWithHandler.Length; i++)
{
var pipeline = pipelinesWithHandler[i];
pipeline.SetNext(lastPipeline);
lastPipeline = pipeline;
}
return lastPipeline;
});
This elegant design chains pipeline behaviors at resolution time — no static lists, no reflection, no magic.
It's simple! Just use the following code:
builder.Services.AddDispatchR(typeof(MyCommand).Assembly, withPipelines: true, withNotifications: true);
This code will automatically register all pipelines by default. If you need to register them in a specific order, you can either add them manually or write your own reflection logic:
builder.Services.AddDispatchR(typeof(MyCommand).Assembly, withPipelines: false, withNotifications: false);
builder.Services.AddScoped<IPipelineBehavior<MyCommand, int>, PipelineBehavior>();
builder.Services.AddScoped<IPipelineBehavior<MyCommand, int>, ValidationBehavior>();
builder.Services.AddScoped<IStreamPipelineBehavior<MyStreamCommand, int>, ValidationBehavior>();
builder.Services.AddScoped<INotificationHandler<Event>, EventHandler>();
- Automatic pipeline and notification registration is enabled by default
- Manual registration allows for custom pipeline or notification ordering
- You can implement custom reflection if needed
dotnet add package DispatchR.Mediator --version 1.2.0
Important
This benchmark was conducted using MediatR version 12.5.0 and the stable release of Mediator Source Generator, version 2.1.7. Version 3 of Mediator Source Generator was excluded due to significantly lower performance.
We welcome contributions to make this package even better! ❤️
- Found a bug? → Open an issue
- Have an idea? → Suggest a feature
- Want to code? → Submit a PR
Let's build something amazing together! 🚀