English | 中文
HandyIpc is an out-of-the-box inter-process communication (IPC) library, similar to WCF for remote method calls, but lighter in comparison, eliminating all the tedious configuration. You only need to read this README from the beginning to the master.
This library provides a high-level RMI (remote method invocation) API. Its underlying communication can be implemented by whatever you like, such as Named Pipe, MMF (memory mapping file) or Socket, and this framework does not care about the specific implementation.
Add the following 3 packages to your project.
Note: HandyIpc.NamedPipe
and HandyIpc.Socket
only need to install one, it is recommended to choose HandyIpc.NamedPipe
, it is faster than HandyIpc.Socket
.
<PackageReference Include="HandyIpc" Version="0.5.2" />
<PackageReference Include="HandyIpc.NamedPipe" Version="0.5.0" />
<PackageReference Include="HandyIpc.Serializer.Json" Version="0.5.0" />
// Declare an interface contains a set of methods that needs to be called remotely,
// and mark it with IpcContractAttribute.
[IpcContract]
// Feature: Supports generic interfaces.
public interface IDemo<T>
{
Task<T> GetDefaultAsync();
double Add(double x, double y);
// Feature: Supports Task/Task<T> async methods.
Task<double> AddAsync(double x, double y);
// Feature: Supports generic methdos.
string GetTypeName<T>();
}
// Implement the IPC contract (interface).
public class Demo<T> : IDemo<T>
{
public Task<T> GetDefaultAsync() => Task.FromResult<T>(default);
public double Add(double x, double y) => x + y;
public Task<double> AddAsync(double x, double y) => Task.FromResult(x + y);
public string GetTypeName<T>() => typeof(T).Name;
}
// Create a ContainerServerBuilder instance to build the IContainerServer instance.
ContainerServerBuilder serverBuilder = new();
serverBuilder
//.UseTcp(IPAddress.Loopback, 10086)
.UseNamedPipe("ec57043f-465c-4766-ae49-b9b1ee9ac571")
.UseJsonSerializer();
serverBuilder
// Generic interfaces that are not monomorphic need to be registered in this form.
.Register(typeof(IDemo<>), typeof(Demo<>))
// Non-generic interfaces or generic interfaces that are already monomorphic can use more elegant extension methods.
.Register<IDemo<string>, Demo<string>>()
.Register<ICalculator, Calculator>();
using var server = serverBuilder.Build();
// Don't forget to start the server.
server.Start();
// server.Stop();
// Create a ContainerClientBuilder instance to build the IContainerClient instance.
ContainerClientBuilder clientBuilder = new();
clientBuilder
//.UseTcp(IPAddress.Loopback, 10086)
.UseNamedPipe("ec57043f-465c-4766-ae49-b9b1ee9ac571")
.UseJsonSerializer();
using var client = clientBuilder.Build();
// Resolve contract instances from the client instance.
var demo1 = client.Resolve<IDemo<string>>();
var demo2 = client.Resolve<IDemo<int>>();
// Using contract instances, they will call an implementation from another process.
var result0 = demo1.Add(16, 26); // 42
var result1 = await demo1.AddAsync(40, 2); // 42
var result2 = demo1.GetTypeName<string>(); // "String"
var result3 = await demo1.GetDefaultAsync(); // null
var result3 = await demo2.GetDefaultAsync(); // 0
- Support for generic interface.
- Support for
Task/Task<T>
return value in interface method. - Support for generic methods (parameter type allow contains nested generic types).
- NOT support for interface inheritance.