本文是“项目框架构建”系列之6,本文介绍如何编写通用主机基础类。
1.为了构建通用主机,我们先创建主机接口IAppHost接口
接口需要有配置项,我们定义为HostConfiguration,比如我们希望用户可以设定他的工作目录,就可以放在这里
接口需要有加载的.json的应用程序配置IConfiguration对象
接口还需要有服务管理器IServiceProvider对象
以及接口需要有日志管理ILoggerManager对象
2.构建主机,需要执行方法,所以我们添加了Build()和Run()两个方法
Build()方法是为了生成主机,这是必要方法。
Run()方法是为了启动应用程序,但这并不是必要的,因为主机生成后,您本身可以随时随地去启动你的程序。
3.生命周期以及事件
像主机这种底层结构,为了兼顾扩展性,我们需要提供周期的事件,以便让用户在启动的时候进行参与。
本程序抛砖引玉,提供了4个事件:
HostConfiguring:正在配置IConfiguration中,您可以继续加入其它初始化工作,如加载其它的.json文件
HostConfigurationInitialized:主机的配置已完成且抽象接口IConfiguration已构建
HostServicesInitializing:主机正在构建服务中,您可以继续加入其它服务一起构建
HostReady:主机构建完成已准备就绪
4.扩展依赖注入
为了防止重复添加服务,我们还需要扩展新增一些方法,用于防止重复添加服务。
5.扩展appsettings.cs文件的内容更改
程序运行后,有时可能需要在运行中修改appsettings.json中的配置,但IConfiguration本身不带有这种功能,所以我们需要自行编写扩展
6.编写主机接口IAppHost的抽象实现
我们是通用框架项目,所以需要提供一个抽象的基础实现,依据之前的设计,我们实现它的方法、事件、属性。
我们限定构造函数
程序初始化,初始化应用程序配置以及基础服务
编写Build()方法实现主机的生成
由于Run()函数不是必须的,所以我们把它定义为虚方法
下面通过编写事件的虚方法,以便子类能够参与过程
最后,附上完整的主机项目结构图:
也许各位朋友会想要源码,不要慌,等系列文章差不多了后,会开放源码下载。现在也没多少代码,没啥看头的。
下面附上IAppHost的具体实现:
using System;
using System.IO;
using Xejen.Hosting.Extensions;
using Xejen.Logger;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;namespace Xejen.Hosting
{/// <summary>/// <inheritdoc cref="IAppHost"/>基类 /// </summary>/// <creator>marc</creator>public abstract class AppHostBase : IAppHost{private readonly ILogger _logger;private readonly ILoggerManager _loggerManager;/// <inheritdoc cref="ILogger"/>protected ILogger Logger => _logger;/// <summary>/// 服务列表/// </summary>internal IServiceCollection Services;/// <inheritdoc cref="HostConfiguration"/>public HostConfiguration HostConfig { get; private set; }/// <inheritdoc/>public IConfiguration Configuration { get; private set; }/// <inheritdoc/>public IServiceProvider ServiceProvider { get; private set; }/// <inheritdoc/>public ILoggerManager LoggerManager => _loggerManager;/// <inheritdoc/>public event EventHandler<IConfigurationBuilder> HostConfiguring;/// <inheritdoc/>public event EventHandler<IConfiguration> HostConfigurationInitialized;/// <inheritdoc/>public event EventHandler<IServiceCollection> HostServicesInitializing;/// <inheritdoc/>public event EventHandler HostReady;/// <inheritdoc cref="AppHostBase"/>/// <param name="args">启动参数</param>/// <param name="loggerManager">采用的日志方式</param>protected AppHostBase(string[] args, ILoggerManager loggerManager) : this(loggerManager){HostConfig = HostConfiguration.Load(args);Initialize();}/// <summary>/// 将 ILoggerManager 添加到构造函数中/// </summary>/// <param name="loggerManager"><inheritdoc cref="ILoggerManager" path="/summary"/></param>private AppHostBase(ILoggerManager loggerManager){Check.NotNull(loggerManager, nameof(loggerManager));_loggerManager = loggerManager;_logger = _loggerManager.CreateLogger(typeof(AppHostBase));}/// <inheritdoc cref="AppHostBase"/>/// <param name="configuration"><inheritdoc cref="HostConfiguration" path="/summary"/></param>/// <param name="loggerManager"><inheritdoc cref="ILoggerManager" path="/summary"/></param>protected AppHostBase(HostConfiguration configuration, ILoggerManager loggerManager) : this(loggerManager){Check.NotNull(configuration, nameof(configuration));HostConfig = configuration;Initialize();}private void Initialize(){_logger.LogInformation($"应用程始启动,开始初始化");InitializeConfiguration(HostConfig.ConfigDirectory);InitializeServices();}/// <inheritdoc/>public IAppHost Build(){ConfigureServices(Services);ServiceProvider = Services.BuildServiceProvider();_logger.LogInformation($"服务初始化完毕,共 {Services.Count} 项服务");_logger.LogInformation($"程序初始化完毕");OnHostReady(this);return this;}/// <inheritdoc/>public virtual void Run() { }private void InitializeConfiguration(string configDirectory){_logger.LogInformation($"准备加载配置文件");if (string.IsNullOrEmpty(configDirectory)){configDirectory = AppDomain.CurrentDomain.BaseDirectory;}string appsettingFileName = HostConfig.DefaultSettingsFileName;if (!File.Exists(Path.Combine(configDirectory, appsettingFileName))){throw new FileNotFoundException($"未找到 {appsettingFileName} 文件,查找目录在:\r\n{configDirectory}");}var builder = new ConfigurationBuilder().SetBasePath(configDirectory).AddJsonFile(appsettingFileName, optional: true, reloadOnChange: true);ConfigureJsonFiles(builder);OnHostConfiguring(this, builder);Configuration = builder.Build();OnHostConfigurationInitialized(this, Configuration);_logger.LogInformation($"配置文件加载完成,工作目录: {configDirectory}");}/// <summary>/// 配置json文件,基类已自动完成<see langword="appsettings.json"/>文件的加载,您只需加载您其它的配置文件即可/// </summary>/// <param name="configuration">传入过来的<see cref="IConfigurationBuilder"/>接口对象</param>protected virtual void ConfigureJsonFiles(IConfigurationBuilder configuration){}/// <summary>/// 初始化基础服务/// </summary>private void InitializeServices(){_logger.LogInformation($"准备初始化服务");var services = new ServiceCollection();Services = services;// 添加服务// services.AddTransient<IService, ServiceImplementation>();// 添加主机服务:public class AppHostedService : IHostedService,该类要实现StartAsync以及StopAsync方法// services.AddHostedService<AppHostedService>();services.AddSingleton(_loggerManager);services.AddSingleton(Configuration);services.AddSingleton(HostConfig);OnHostServicesInitializing(this, services);}/// <summary>/// 配置服务,各子项目可以重写此方法来注册各项服务/// </summary>/// <param name="services">服务集合</param>protected virtual void ConfigureServices(IServiceCollection services){}/// <inheritdoc cref="HostConfiguring" path="/summary"/>/// <param name="sender"><inheritdoc cref="IAppHost" path="/summary"/></param>/// <param name="builder">用于构建应用程序配置的构建器</param>protected virtual void OnHostConfiguring(IAppHost sender, IConfigurationBuilder builder){HostConfiguring?.Invoke(sender, builder);}/// <inheritdoc cref="HostConfigurationInitialized" path="/summary"/>/// <param name="sender"><inheritdoc cref="IAppHost" path="/summary"/></param>/// <param name="builder">应用程序配置</param>protected virtual void OnHostConfigurationInitialized(IAppHost sender, IConfiguration builder){HostConfigurationInitialized?.Invoke(sender, builder);}/// <inheritdoc cref="HostServicesInitializing" path="/summary"/>/// <param name="sender"><inheritdoc cref="IAppHost" path="/summary"/></param>/// <param name="builder">服务列表</param>protected virtual void OnHostServicesInitializing(IAppHost sender, IServiceCollection builder){HostServicesInitializing?.Invoke(sender, builder);}/// <inheritdoc cref="HostReady" path="/summary"/>/// <param name="sender"><inheritdoc cref="IAppHost" path="/summary"/></param>protected virtual void OnHostReady(IAppHost sender){HostReady?.Invoke(sender, EventArgs.Empty);}}
}
祝您用餐愉快,下一篇我们将编写如何使用主机框架的实际应用,敬请期待
1-3-5 $ 3-5-5-4 带着田螺回四堡 3-5-2-4