项目框架构建之6:编写通用主机基础类

本文是“项目框架构建”系列之6,本文介绍如何编写通用主机基础类。

1.为了构建通用主机,我们先创建主机接口IAppHost接口

接口需要有配置项,我们定义为HostConfiguration,比如我们希望用户可以设定他的工作目录,就可以放在这里

项目框架构建之6:编写通用主机基础类_第1张图片


接口需要有加载的.json的应用程序配置IConfiguration对象
接口还需要有服务管理器IServiceProvider对象
以及接口需要有日志管理ILoggerManager对象

项目框架构建之6:编写通用主机基础类_第2张图片


2.构建主机,需要执行方法,所以我们添加了Build()和Run()两个方法
Build()方法是为了生成主机,这是必要方法。
Run()方法是为了启动应用程序,但这并不是必要的,因为主机生成后,您本身可以随时随地去启动你的程序。

项目框架构建之6:编写通用主机基础类_第3张图片


3.生命周期以及事件

像主机这种底层结构,为了兼顾扩展性,我们需要提供周期的事件,以便让用户在启动的时候进行参与。


本程序抛砖引玉,提供了4个事件:
HostConfiguring:正在配置IConfiguration中,您可以继续加入其它初始化工作,如加载其它的.json文件
HostConfigurationInitialized:主机的配置已完成且抽象接口IConfiguration已构建
HostServicesInitializing:主机正在构建服务中,您可以继续加入其它服务一起构建
HostReady:主机构建完成已准备就绪

项目框架构建之6:编写通用主机基础类_第4张图片


4.扩展依赖注入

为了防止重复添加服务,我们还需要扩展新增一些方法,用于防止重复添加服务。

项目框架构建之6:编写通用主机基础类_第5张图片


5.扩展appsettings.cs文件的内容更改

程序运行后,有时可能需要在运行中修改appsettings.json中的配置,但IConfiguration本身不带有这种功能,所以我们需要自行编写扩展

项目框架构建之6:编写通用主机基础类_第6张图片


6.编写主机接口IAppHost的抽象实现

我们是通用框架项目,所以需要提供一个抽象的基础实现,依据之前的设计,我们实现它的方法、事件、属性。

我们限定构造函数

项目框架构建之6:编写通用主机基础类_第7张图片


程序初始化,初始化应用程序配置以及基础服务

项目框架构建之6:编写通用主机基础类_第8张图片


编写Build()方法实现主机的生成

项目框架构建之6:编写通用主机基础类_第9张图片


由于Run()函数不是必须的,所以我们把它定义为虚方法


 

下面通过编写事件的虚方法,以便子类能够参与过程

项目框架构建之6:编写通用主机基础类_第10张图片

最后,附上完整的主机项目结构图:

项目框架构建之6:编写通用主机基础类_第11张图片

也许各位朋友会想要源码,不要慌,等系列文章差不多了后,会开放源码下载。现在也没多少代码,没啥看头的。

下面附上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
{
    /// 
    /// 基类 
    /// 
    /// marc
    public abstract class AppHostBase : IAppHost
    {
        private readonly ILogger _logger;
        private readonly ILoggerManager _loggerManager;
        /// 
        protected ILogger Logger => _logger;

        /// 
        /// 服务列表
        /// 
        internal IServiceCollection Services;

        /// 
        public HostConfiguration HostConfig { get; private set; }
        /// 
        public IConfiguration Configuration { get; private set; }
        /// 
        public IServiceProvider ServiceProvider { get; private set; }
        /// 
        public ILoggerManager LoggerManager => _loggerManager;

        /// 
        public event EventHandler HostConfiguring;
        /// 
        public event EventHandler HostConfigurationInitialized;
        /// 
        public event EventHandler HostServicesInitializing;
        /// 
        public event EventHandler HostReady;

        /// 
        /// 启动参数
        /// 采用的日志方式
        protected AppHostBase(string[] args, ILoggerManager loggerManager) : this(loggerManager)
        {
            HostConfig = HostConfiguration.Load(args);

            Initialize();
        }

        /// 
        /// 将 ILoggerManager 添加到构造函数中
        /// 
        /// 
        private AppHostBase(ILoggerManager loggerManager)
        {
            Check.NotNull(loggerManager, nameof(loggerManager));

            _loggerManager = loggerManager;
            _logger = _loggerManager.CreateLogger(typeof(AppHostBase));
        }

        /// 
        /// 
        /// 
        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();
        }

        /// 
        public IAppHost Build()
        {
            ConfigureServices(Services);

            ServiceProvider = Services.BuildServiceProvider();

            _logger.LogInformation($"服务初始化完毕,共 {Services.Count} 项服务");

            _logger.LogInformation($"程序初始化完毕");

            OnHostReady(this);

            return this;
        }

        /// 
        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}");
        }

        /// 
        /// 配置json文件,基类已自动完成文件的加载,您只需加载您其它的配置文件即可
        /// 
        /// 传入过来的接口对象
        protected virtual void ConfigureJsonFiles(IConfigurationBuilder configuration)
        {

        }

        /// 
        /// 初始化基础服务
        /// 
        private void InitializeServices()
        {
            _logger.LogInformation($"准备初始化服务");
            var services = new ServiceCollection();
            Services = services;

            // 添加服务
            // services.AddTransient();
            // 添加主机服务:public class AppHostedService : IHostedService,该类要实现StartAsync以及StopAsync方法
            // services.AddHostedService();

            services.AddSingleton(_loggerManager);
            services.AddSingleton(Configuration);
            services.AddSingleton(HostConfig);

            OnHostServicesInitializing(this, services);
        }

        /// 
        /// 配置服务,各子项目可以重写此方法来注册各项服务
        /// 
        /// 服务集合
        protected virtual void ConfigureServices(IServiceCollection services)
        {

        }

        /// 
        /// 
        /// 用于构建应用程序配置的构建器
        protected virtual void OnHostConfiguring(IAppHost sender, IConfigurationBuilder builder)
        {
            HostConfiguring?.Invoke(sender, builder);
        }

        /// 
        /// 
        /// 应用程序配置
        protected virtual void OnHostConfigurationInitialized(IAppHost sender, IConfiguration builder)
        {
            HostConfigurationInitialized?.Invoke(sender, builder);
        }

        /// 
        /// 
        /// 服务列表
        protected virtual void OnHostServicesInitializing(IAppHost sender, IServiceCollection builder)
        {
            HostServicesInitializing?.Invoke(sender, builder);
        }

        /// 
        /// 
        protected virtual void OnHostReady(IAppHost sender)
        {
            HostReady?.Invoke(sender, EventArgs.Empty);
        }
    }
}

祝您用餐愉快,下一篇我们将编写如何使用主机框架的实际应用,敬请期待

1-3-5 $ 3-5-5-4 带着田螺回四堡 3-5-2-4

你可能感兴趣的:(项目框架实战,框架结构,Host主机,C#,面向对象)