天天看點

使用Topshelf元件建構簡單的Windows服務

   很多時候都在讨論是否需要了解一個元件或者一個語言的底層原理這個問題,其實我個人覺得,對于這個問題,每個人都有自己的看法,個人情況不同,選擇的方式也就會不同了。我個人覺得無論學習什麼,都應該嘗試着去了解對應的原理和源碼(這裡就不要急着吐槽,容我說完)。對底層的了解不是為了讓你寫出類似的東西,讓你寫也不可能寫的出來,重寫一個就需要以此修改整個底層結構,了解底層知識隻是為了讓你可以在寫業務代碼時,選擇合适的方式,以此使底層與業務層配合達到效率最佳。任何一種方式有壞有好,需要合适的選擇。

  如果覺得樓主以上的說法不對,或者有些不妥,還望見諒,因為争論一個觀點沒有意義,認為對的人自己會去了解,認為不對的,可以忽略。沒有這個必要去花費時間和精力取讨論這種事情。

  以上是扯淡,下面切入正題。前面介紹了一個元件Hangfire,用于設定定時任務等等操作,在這裡介紹另一款元件Topshelf。

一.Topshelf元件概述

   Topshelf是.NET平台的Windows服務架構。Topshelf可以輕松建立Windows服務,測試服務,調試服務,并最終将其安裝到Windows服務控制管理器(SCM)中。Topshelf通過允許開發人員專注于服務邏輯,而不是與.NET架構中的内置服務支援互動的細節。開發人員不需要了解服務類的複雜細節,通過InstallUtil執行安裝,或者了解如何将調試器附加到服務以進行故障排除問題。

   建立Windows服務與建立控制台應用程式類似,控制台應用程式建立後,建立一個具有公共Start和Stop方法的單一服務類。服務操作的方式較多,自動,自動(延遲),手動和禁用啟動選項本地系統,本地服務,網絡服務,使用者名/密碼或安裝期間提示的服務憑證。服務啟動依賴項,包括SQL Server,MSMQ和其他具有不同服務名稱的多執行個體服務安裝服務恢複選項,包括重新啟動,重新開機或運作程式。Topshelf與Mono合作,可以将服務部署到Linux。服務安裝功能目前僅限Windows。

二.Topshelf用法說明

      介紹完對應的元件背景概述,在這裡就要介紹一下如何使用這個元件的使用方法。該元件的使用方法有另個方法,都在HostFactory類中,下面具體的介紹一個使用方式。

   1.配置新的服務主機

使用Topshelf元件建構簡單的Windows服務
HostFactory.New(x =>
                {
                    // 可以定義不需要接口依賴性的服務,這隻是為了
                    //在此示例中顯示并未使用。
                    x.Service<SampleSansInterfaceService>(s =>
                        {
                            s.ConstructUsing(() => new SampleSansInterfaceService());
                            s.WhenStarted(v => v.Start());
                            s.WhenStopped(v => v.Stop());
                        });
                });      
使用Topshelf元件建構簡單的Windows服務

   2.配置和運作新的服務主機,處理任何異常并将其寫入日志

使用Topshelf元件建構簡單的Windows服務
HostFactory.Run(x =>
                {
                    x.UseLog4Net("log4net.config");
                    x.UseAssemblyInfoForServiceInfo();
                    bool throwOnStart = false;
                    bool throwOnStop = false;
                    bool throwUnhandled = false;
                    x.Service(settings => new SampleService(throwOnStart, throwOnStop, throwUnhandled), s =>
                    {
                        s.BeforeStartingService(_ => Console.WriteLine("BeforeStart"));
                        s.BeforeStoppingService(_ => Console.WriteLine("BeforeStop"));
                    });
                    x.SetStartTimeout(TimeSpan.FromSeconds(10));
                    x.SetStopTimeout(TimeSpan.FromSeconds(10));
                    x.EnableServiceRecovery(r =>
                        {
                            r.RestartService(3);
                            r.RunProgram(7, "ping google.com");
                            r.RestartComputer(5, "message");

                            r.OnCrashOnly();
                            r.SetResetPeriod(2);
                        });
                    x.AddCommandLineSwitch("throwonstart", v => throwOnStart = v);
                    x.AddCommandLineSwitch("throwonstop", v => throwOnStop = v);
                    x.AddCommandLineSwitch("throwunhandled", v => throwUnhandled = v);
                    x.OnException((exception) =>
                    {
                        Console.WriteLine("Exception thrown - " + exception.Message);
                    });
                });      
使用Topshelf元件建構簡單的Windows服務

    3.Topshelf配置操作方法

使用Topshelf元件建構簡單的Windows服務

三.Topshelf核心對象解析

     承接上文,介紹完畢相關背景和正常操作,在這裡介紹一個核心對象的一些方法。

  1.HostFactory.New():

使用Topshelf元件建構簡單的Windows服務
public static Host New(Action<HostConfigurator> configureCallback)
        {
            try
            {
                if (configureCallback == null)
                    throw new ArgumentNullException("configureCallback");
                var configurator = new HostConfiguratorImpl();
                Type declaringType = configureCallback.Method.DeclaringType;
                if (declaringType != null)
                {
                    string defaultServiceName = declaringType.Namespace;
                    if (!string.IsNullOrEmpty(defaultServiceName))
                        configurator.SetServiceName(defaultServiceName);
                }
                configureCallback(configurator);
                configurator.ApplyCommandLine();
                ConfigurationResult result = ValidateConfigurationResult.CompileResults(configurator.Validate());
                if (result.Message.Length > 0)
                {
                    HostLogger.Get(typeof(HostFactory))
                              .InfoFormat("Configuration Result:\n{0}", result.Message);
                }
                return configurator.CreateHost();
            }
            catch (Exception ex)
            {
                HostLogger.Get(typeof(HostFactory)).Error("An exception occurred creating the host", ex);
                HostLogger.Shutdown();
                throw;
            }
        }      
使用Topshelf元件建構簡單的Windows服務

     該方法用于配置新的服務主機,方法接受一個參數Action<HostConfigurator>配置方法調用,該方法傳回Host對象,表示Topshelf服務主機,準備運作。 configureCallback.Method.DeclaringType;用于擷取聲明該成員的類。declaringType.Namespace;用于擷取擷取 System.Type 的命名空間。ValidateConfigurationResult.CompileResults(configurator.Validate());用于驗證配置結果。

   2.HostFactory.Run():

使用Topshelf元件建構簡單的Windows服務
public static TopshelfExitCode Run(Action<HostConfigurator> configureCallback)
        {
            try
            {
                return New(configureCallback)
                    .Run();
            }
            catch (Exception ex)
            {
                HostLogger.Get(typeof(HostFactory))
                          .Error("The service terminated abnormally", ex);
                HostLogger.Shutdown();
                
                return TopshelfExitCode.AbnormalExit;
            }
        }      
使用Topshelf元件建構簡單的Windows服務

     該方法是一個靜态方法,配置和運作新的服務主機,處理任何異常并将其寫入日志。該方法接收一個參數Action<HostConfigurator> configureCallback配置方法調用,傳回應用程式主方法傳回的程序的退出代碼。

四.總結

   以上是介紹如何使用Topshelf元件建立簡單的Windows服務的方法,在這裡隻是一個簡單的介紹,沒有很深入的介紹,如果需要了解更多的東西,可以看源碼,畢竟是開源免費的元件,也是一個很不錯的元件。