C#創建、安裝、卸載、調試Windows Service(Windows 服務)的簡單教程

来源:http://www.cnblogs.com/Scl891004X/archive/2017/01/02/6243515.html
-Advertisement-
Play Games

本文向大家介紹瞭如何運用C#來創建、安裝、卸載、調試Windows Service程式。有需要的可以參考下,希望對大家有所幫助。 ...


前言:Microsoft Windows 服務能夠創建在它們自己的 Windows 會話中可長時間運行的可執行應用程式。這些服務可以在電腦啟動時自動啟動,可以暫停和重新啟動而且不顯示任何用戶界面。這使服務非常適合在伺服器上使用,或任何時候,為了不影響在同一臺電腦上工作的其他用戶,需要長時間運行功能時使用。還可以在不同於登錄用戶的特定用戶帳戶或預設電腦帳戶的安全上下文中運行服務。本文就向大家介紹如何運用C#來創建、安裝、卸載、調試Windows Service程式。

一、創建Windows服務

1)用VS新建Windows 服務項目

2)預設生成文件包括Program.cs,Service1.cs。重命名Service1.cs為你的服務名或刪除Service1.cs文件然後創建自己的服務文件,假設取服務名字為MyService。註意:如果是刪除Service1.cs文件然後創建自己的服務文件,需要將Program.cs文件里的Service1修改為MyService。

MyService.cs屬性視窗中,相關屬性如下:

Autolog                       是否自動寫入系統的日誌文件
CanHandlePowerEvent       服務時候接受電源事件
CanPauseAndContinue         服務是否接受暫停或繼續運行的請求
CanShutdown        服務是否在運行它的電腦關閉時收到通知,以便能夠調用 OnShutDown 過程
CanStop                              服務是否接受停止運行的請求
ServiceName                       服務名

註意:CanPauseAndContinue和CanShutdown的預設值均為False,要想使服務的OnPause()、OnContinue()、OnShutdown()起作用,需要將CanPauseAndContinue和CanShutdown屬性值設置為True。

3)雙擊MyService.cs服務文件,在左側設計模式中,右鍵點擊“添加安裝程式”(或者在MyService.cs的屬性視窗的下方點擊添加“添加安裝程式”;如果看不到“添加安裝程式”的可點鏈接,可以右鍵屬性視窗,點擊“命令(C)”後就會出來了。註意:是屬性視窗而不是文件屬性視窗),會自動生成Projectinstaller.cs文件以及兩個安裝組件,如下:

4)單擊“serviceProcessInstaller1”,在其屬性視窗中設置Account帳號方式,建議為LocalService(當然也可以Account屬性改為 LocalSystem,這樣,不論是以哪個用戶登錄的系統,服務總會啟動)。

5)單擊“serviceInstaller1”,在其屬性視窗設置屬性:

  a)Description 服務描述,直接顯示到Windows服務列表中的描述;

  b)DisplayName 服務顯示名稱,直接顯示到Windows服務列表中的名稱;

  c)ServiceName 服務進程名稱,安裝與卸載服務時的唯一標識。

具體設置如上圖所示。

6)創建安裝服務批處理文件Install.bat,可以創建記事本,然後修改尾碼為bat,記事本內容如下:

%SystemRoot%\Microsoft.NET\Framework\v4.0.30319\installutil.exe WindowsServiceDemo.exe
Net Start MyService
sc config MyService start= auto
pause

註意:記事本另存為時設置編碼為ANSI

說明:第二行為啟動服務,第三行為設置服務為自動運行,這兩行視服務形式自行選擇。如果需要查看腳本運行狀況,在腳本最後一行加入pause。

7)同理創建卸載服務批處理文件Uninstall.bat,內容如下:

%SystemRoot%\Microsoft.NET\Framework\v4.0.30319\installutil.exe /u WindowsServiceDemo.exe
pause

8)將Install.bat以及Uninstall.bat這兩個文件添加到bin\Debug目錄下,此時解決方案的目錄結構如下:

 

9)寫服務代碼,以向文本文件寫入文本記錄系統時間為例:

  1 using System;
  2 using System.IO;
  3 using System.Diagnostics;
  4 using System.ServiceProcess;
  5 using System.Timers;
  6 
  7 namespace WindowsServiceDemo
  8 {
  9     public partial class MyService : ServiceBase
 10     {
 11         private Timer time = new Timer();
 12         public MyService()
 13         {
 14             InitializeComponent();
 15         }
 16 
 17         protected override void OnStart(string[] args)
 18         {
 19             #if DEBUG
 20             if (!Debugger.IsAttached)
 21                 Debugger.Launch();
 22             Debugger.Break();
 23             #endif
 24             WriteLog("服務啟動,時間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n");
 25             time.Elapsed += new ElapsedEventHandler(MethodEvent);
 26             time.Interval = 60 * 1000;//時間間隔為2秒鐘
 27             time.Start();
 28         }
 29 
 30         protected override void OnStop()
 31         {
 32             #if DEBUG
 33             if (!Debugger.IsAttached)
 34                 Debugger.Launch();
 35             Debugger.Break();
 36             #endif
 37             WriteLog("服務停止,時間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n");
 38         }
 39 
 40         protected override void OnPause()
 41         {
 42             #if DEBUG
 43             if (!Debugger.IsAttached)
 44                 Debugger.Launch();
 45             Debugger.Break();
 46             #endif
 47             WriteLog("服務暫停,時間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n");
 48             base.OnPause();
 49         }
 50 
 51         protected override void OnContinue()
 52         {
 53             #if DEBUG
 54             if (!Debugger.IsAttached)
 55                 Debugger.Launch();
 56             Debugger.Break();
 57             #endif
 58             WriteLog("服務恢復,時間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n");
 59             base.OnContinue();
 60         }
 61 
 62         protected override void OnShutdown()
 63         {
 64             WriteLog("電腦關閉,時間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n");
 65             base.OnShutdown();
 66         }
 67 
 68         private void MethodEvent(object source, System.Timers.ElapsedEventArgs e)
 69         {
 70             time.Enabled = false;
 71             string result = string.Empty;
 72             try
 73             {
 74                 //.........
 75                 result = "執行成功,時間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n";
 76             }
 77             catch (Exception ex)
 78             {
 79                 result = "執行失敗,原因:" + ex.Message + "\r\n";
 80             }
 81             finally
 82             {
 83                 WriteLog(result);
 84                 time.Enabled = true;
 85             }
 86         }
 87         /// <summary>
 88         /// 日誌記錄
 89         /// </summary>
 90         /// <param name="logInfo"></param>
 91         private void WriteLog(string logInfo)
 92         {
 93             try
 94             {
 95                 string logDirectory = AppDomain.CurrentDomain.BaseDirectory + "\\Logs";
 96                 if (!Directory.Exists(logDirectory))
 97                 {
 98                     Directory.CreateDirectory(logDirectory);
 99                 }
100                 string filePath = logDirectory + "\\" + DateTime.Now.ToString("yyyy-MM-dd") + ".txt";
101                 File.AppendAllText(filePath, logInfo);
102             }
103             catch
104             { 
105 
106             }
107         }
108     }
109 }
向文本文件寫入文本記錄系統時間

註意:代碼編寫完成後,你無法點擊通過啟動按鈕或按F5來運行或調試服務,會彈出如下圖所示的警告:

二、安裝windows服務

項目生成成功後,定位到bin\Debug目錄,以管理員身份運行Install.bat安裝服務,成功結果如下圖:

 

這時,“我的電腦”右鍵,選擇“管理”,選擇“服務和應用程式”下的“服務”,就可以看到服務已安裝,如下圖:

同時,Debug文件夾里有了Logs文件夾,Logs文件夾里有txt文檔,內容如下:

可以看到,每分鐘執行一次。

三、調試windows服務

1)通常的處理辦法是,在service運行後, 在調試器中選擇“附加到進程”,附加自己的服務即可調試。但此法有局限性,例如在service啟動時的OnStart事件中的代碼, 基本上很難調試,往往當attach到我們的service的時候,這部分代碼已經執行過了。當然了,你可以讓OnStart事件之前先睡個20s,趁著服務睡覺的時候趕緊“附加到進程”。 System.Threading.Thread.Sleep(1000 * 20); 

2)我的做法是,在OnStart事件的最開始部分加上“Debugger.Launch();”的調用, 當service運行到此處時,將會彈出一個選擇調試器的對話框,同時暫停在當前位置。這樣,我們就做到了在代碼中手動的啟動調試器。

說明:a)Debugger.Launch()方法的作用是“啟動調試器並將其連接到進程”;

   b)可以手動設置斷點,也可以用“Debugger.Break();”動態設置斷點;

   c)為了避免多個調試器實例,可以用“Debugger.IsAttached”屬性判斷調試器是否已附加到進程,代碼片段: if (!Debugger.IsAttached) Debugger.Launch(); 

   d)為了使調試只在Debug模式下生效,Release模式下無效,可以用條件編譯來處理,代碼片段如下:

 #if DEBUG
 if (!Debugger.IsAttached)
     Debugger.Launch();
 Debugger.Break();
 #endif

   關於條件編譯,請查看我的另一篇博客:C#-#define條件編譯

   e)在調試服務的其他事件或方法時,同樣可以用到。

彈出選擇調試器的對話框,以及調試界面如下圖所示:

 

四、卸載windows服務

卸載服務,同樣以管理員身份運行Uninstall.bat即可,成功結果如下圖:

參考鏈接:https://msdn.microsoft.com/zh-cn/library/windows/desktop/system.diagnostics.debugger(v=vs.110).aspx

源碼下載:WindowsServiceDemo.rar


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在接著寫Asp.Net WebApi核心對象解析(下篇)之前,還是一如既往的扯扯淡,元旦剛過,整個人還是處於暈的狀態,一大早就來處理系統BUG,簡直是坑爹(好在沒讓我元旦趕過來該BUG),隊友挖的坑,還讓我含著淚去填。改BUG前看類看隊友寫的代碼,這裡就不評價了,反正是邊改邊罵,我的嘴巴就沒停過,作 ...
  • 要想學會對象編程,首先要掌握面向對象編程的原則,方法,模式和技巧,並遵循它們來寫程式。這樣程式健壯,易於擴展,易維護。 面向對象編程的原則: 單一職責原則:一個對象要專註的做一件事; 開放封閉原則:開放擴展,封閉修改;(生成文件中,如果修改只要修改裡面的dll文件就可以了,不需要重新編譯生成安裝等) ...
  • 註: 調用是可使用Word.Application wdApp = GetWordApplication(true); 獲取該進程中的所有文檔可用:Word.Documents docs = wdApp.Documents; 轉載請註明出處,謝謝! ...
  • 在用C#開發Web應用時有個痛點,就是本機用VS開啟Web應用調試時外部機器無法訪問此Web應用。這裡將會介紹如何通過設置允許區域網和外網機器訪問本機的Web應用。 ...
  • 在一次請求中,即一個線程內,若是用到EF數據上下文對象,就創建一個,這也加是很多人的代碼中習慣在使用上下文對象時,習慣將對象建立在using中,也是為了儘早釋放上下文對象, 但是如果有一個業務邏輯調用了多個dal層的方法,交互資料庫多次,這樣效率會低一些,而且在使用EF的情況下,我們通常把SaveC ...
  • 通過ajax跨域方式調用WCF服務,實現小票pos機的列印,源碼提供web方式,客戶端方式測試,服務駐留右側底部任務欄,可控制服務開啟暫停,用戶可自定義小票列印模板,配合零售錄入。 qq 2294508843 ...
  • 代碼: using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Web; namespace Common.Utils { /// <summary> /// ...
  • 前言 2017年就這麼悄無聲息的開始了,2017年對我來說又是特別重要的一年。 元旦放假在家寫了個Asp.net Core驗證碼登錄, 做demo的過程中遇到兩個小問題,第一是在Asp.net Core中引用dll,以往我們引用DLL都是直接引用,在Core里這樣是不行的,必須基於NuGet添加,或 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...