您现在可以使用 Microsoft.Extensions.Hosting pacakge,它是 asp.net 核心和控制台应用程序的托管和启动基础架构。
与 asp.net core 一样,您可以使用 HostBuilder API 开始构建 gRPC 主机并进行设置。以下代码是为了让控制台应用程序一直运行直到它停止(例如使用 Control-C):
using System;
using System.Threading.Tasks;
using Microsoft.Extensions.Hosting;
public class Program
{
public static async Task Main(string[] args)
{
var hostBuilder = new HostBuilder();
// register your configuration and services.
....
await hostBuilder.RunConsoleAsync();
}
}
为了运行 gRPC 服务,您需要在托管服务中启动/停止 Grpc.Core.Server。托管服务基本上是在主机本身启动时由主机运行的一段代码,在它停止时也是如此。这在 IHostedService 接口中表示。也就是说,实现一个 GrpcHostedService 来覆盖接口:
using System.Threading;
using System.Threading.Tasks;
using Grpc.Core;
using Microsoft.Extensions.Hosting;
namespace Grpc.Host
{
public class GrpcHostedService: IHostedService
{
private Server _server;
public GrpcHostedService(Server server)
{
_server = server;
}
public Task StartAsync(CancellationToken cancellationToken)
{
_server.Start();
return Task.CompletedTask;
}
public async Task StopAsync(CancellationToken cancellationToken) => await _server.ShutdownAsync();
}
}
真的很简单。我们通过依赖注入获得一个GrpcHostedService 实例,并在主机启动时在其上运行 StartAsync。当主机停止时,我们运行 StopAsync 以便我们可以优雅地关闭所有东西,包括 Grpc 服务器。
然后回到Program.cs并进行一些更改:
public class Program
{
public static async Task Main(string[] args)
{
var hostBuilder = new HostBuilder()
// Add configuration, logging, ...
.ConfigureServices((hostContext, services) =>
{
// Better to use Dependency Injection for GreeterImpl
Server server = new Server
{
Services = {Greeter.BindService(new GreeterImpl())},
Ports = {new ServerPort("localhost", 5000, ServerCredentials.Insecure)}
};
services.AddSingleton<Server>(server);
services.AddSingleton<IHostedService, GrpcHostedService>();
});
await hostBuilder.RunConsoleAsync();
}
}
通过这样做,通用主机将自动在我们的托管服务上运行 StartAsync,这反过来又会在 Server 实例上调用 StartAsync,实质上是启动 gRPC 服务器。
当我们使用 Control-C 关闭主机时,通用主机将自动在我们的托管服务上调用 StopAsync,该服务将再次在 Server 实例上调用 StopAsync,这将进行一些清理。
HostBuilder中的其他配置可以看这个blog。