C#使用Windows Service的簡(jiǎn)單教程(創(chuàng)建、安裝、卸載、調(diào)試)
前言:Microsoft Windows 服務(wù)能夠創(chuàng)建在它們自己的 Windows 會(huì)話中可長(zhǎng)時(shí)間運(yùn)行的可執(zhí)行應(yīng)用程序。這些服務(wù)可以在計(jì)算機(jī)啟動(dòng)時(shí)自動(dòng)啟動(dòng),可以暫停和重新啟動(dòng)而且不顯示任何用戶界面。這使服務(wù)非常適合在服務(wù)器上使用,或任何時(shí)候,為了不影響在同一臺(tái)計(jì)算機(jī)上工作的其他用戶,需要長(zhǎng)時(shí)間運(yùn)行功能時(shí)使用。還可以在不同于登錄用戶的特定用戶帳戶或默認(rèn)計(jì)算機(jī)帳戶的安全上下文中運(yùn)行服務(wù)。本文就向大家介紹如何運(yùn)用C#來(lái)創(chuàng)建、安裝、卸載、調(diào)試Windows Service程序。
一、創(chuàng)建Windows服務(wù)
1)用VS新建Windows 服務(wù)項(xiàng)目
2)默認(rèn)生成文件包括Program.cs,Service1.cs。重命名Service1.cs為你的服務(wù)名或刪除Service1.cs文件然后創(chuàng)建自己的服務(wù)文件,假設(shè)取服務(wù)名字為MyService。注意:如果是刪除Service1.cs文件然后創(chuàng)建自己的服務(wù)文件,需要將Program.cs文件里的Service1修改為MyService。
MyService.cs屬性窗口中,相關(guān)屬性如下:
Autolog 是否自動(dòng)寫入系統(tǒng)的日志文件
CanHandlePowerEvent 服務(wù)時(shí)候接受電源事件
CanPauseAndContinue 服務(wù)是否接受暫?;蚶^續(xù)運(yùn)行的請(qǐng)求
CanShutdown 服務(wù)是否在運(yùn)行它的計(jì)算機(jī)關(guān)閉時(shí)收到通知,以便能夠調(diào)用 OnShutDown 過(guò)程
CanStop 服務(wù)是否接受停止運(yùn)行的請(qǐng)求
ServiceName 服務(wù)名
注意:CanPauseAndContinue和CanShutdown的默認(rèn)值均為False,要想使服務(wù)的OnPause()、OnContinue()、OnShutdown()起作用,需要將CanPauseAndContinue和CanShutdown屬性值設(shè)置為True。
3)雙擊MyService.cs服務(wù)文件,在左側(cè)設(shè)計(jì)模式中,右鍵點(diǎn)擊“添加安裝程序”(或者在MyService.cs的屬性窗口的下方點(diǎn)擊添加“添加安裝程序”;如果看不到“添加安裝程序”的可點(diǎn)鏈接,可以右鍵屬性窗口,點(diǎn)擊“命令(C)”后就會(huì)出來(lái)了。注意:是屬性窗口而不是文件屬性窗口),會(huì)自動(dòng)生成Projectinstaller.cs文件以及兩個(gè)安裝組件,如下:
4)單擊“serviceProcessInstaller1”,在其屬性窗口中設(shè)置Account帳號(hào)方式,建議為L(zhǎng)ocalService(當(dāng)然也可以Account屬性改為 LocalSystem,這樣,不論是以哪個(gè)用戶登錄的系統(tǒng),服務(wù)總會(huì)啟動(dòng))。
5)單擊“serviceInstaller1”,在其屬性窗口設(shè)置屬性:
a)Description 服務(wù)描述,直接顯示到Windows服務(wù)列表中的描述;
b)DisplayName 服務(wù)顯示名稱,直接顯示到Windows服務(wù)列表中的名稱;
c)ServiceName 服務(wù)進(jìn)程名稱,安裝與卸載服務(wù)時(shí)的唯一標(biāo)識(shí)。
具體設(shè)置如上圖所示。
6)創(chuàng)建安裝服務(wù)批處理文件Install.bat,可以創(chuàng)建記事本,然后修改后綴為bat,記事本內(nèi)容如下:
%SystemRoot%\Microsoft.NET\Framework\v4.0.30319\installutil.exeWindowsServiceDemo.exe
Net StartMyService
sc config MyServicestart= auto
pause
注意:記事本另存為時(shí)設(shè)置編碼為ANSI
說(shuō)明:第二行為啟動(dòng)服務(wù),第三行為設(shè)置服務(wù)為自動(dòng)運(yùn)行,這兩行視服務(wù)形式自行選擇。如果需要查看腳本運(yùn)行狀況,在腳本最后一行加入pause。
7)同理創(chuàng)建卸載服務(wù)批處理文件Uninstall.bat,內(nèi)容如下:
%SystemRoot%\Microsoft.NET\Framework\v4.0.30319\installutil.exe /uWindowsServiceDemo.exe
pause
8)將Install.bat以及Uninstall.bat這兩個(gè)文件添加到bin\Debug目錄下,此時(shí)解決方案的目錄結(jié)構(gòu)如下:
9)寫服務(wù)代碼,以向文本文件寫入文本記錄系統(tǒng)時(shí)間為例:
using System; using System.IO; using System.Diagnostics; using System.ServiceProcess; using System.Timers; namespace WindowsServiceDemo { public partial class MyService : ServiceBase { private Timer time = new Timer(); public MyService() { InitializeComponent(); } protected override void OnStart(string[] args) { #if DEBUG if (!Debugger.IsAttached) Debugger.Launch(); Debugger.Break(); #endif WriteLog("服務(wù)啟動(dòng),時(shí)間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n"); time.Elapsed += new ElapsedEventHandler(MethodEvent); time.Interval = 60 * 1000;//時(shí)間間隔為2秒鐘 time.Start(); } protected override void OnStop() { #if DEBUG if (!Debugger.IsAttached) Debugger.Launch(); Debugger.Break(); #endif WriteLog("服務(wù)停止,時(shí)間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n"); } protected override void OnPause() { #if DEBUG if (!Debugger.IsAttached) Debugger.Launch(); Debugger.Break(); #endif WriteLog("服務(wù)暫停,時(shí)間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n"); base.OnPause(); } protected override void OnContinue() { #if DEBUG if (!Debugger.IsAttached) Debugger.Launch(); Debugger.Break(); #endif WriteLog("服務(wù)恢復(fù),時(shí)間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n"); base.OnContinue(); } protected override void OnShutdown() { WriteLog("計(jì)算機(jī)關(guān)閉,時(shí)間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n"); base.OnShutdown(); } private void MethodEvent(object source, System.Timers.ElapsedEventArgs e) { time.Enabled = false; string result = string.Empty; try { //......... result = "執(zhí)行成功,時(shí)間:" + DateTime.Now.ToString("HH:mm:ss") + "\r\n"; } catch (Exception ex) { result = "執(zhí)行失敗,原因:" + ex.Message + "\r\n"; } finally { WriteLog(result); time.Enabled = true; } } /// <summary> /// 日志記錄 /// </summary> /// <param name="logInfo"></param> private void WriteLog(string logInfo) { try { string logDirectory = AppDomain.CurrentDomain.BaseDirectory + "\\Logs"; if (!Directory.Exists(logDirectory)) { Directory.CreateDirectory(logDirectory); } string filePath = logDirectory + "\\" + DateTime.Now.ToString("yyyy-MM-dd") + ".txt"; File.AppendAllText(filePath, logInfo); } catch { } } } }
注意:代碼編寫完成后,你無(wú)法點(diǎn)擊通過(guò)啟動(dòng)按鈕或按F5來(lái)運(yùn)行或調(diào)試服務(wù),會(huì)彈出如下圖所示的警告:
二、安裝windows服務(wù)
項(xiàng)目生成成功后,定位到bin\Debug目錄,以管理員身份運(yùn)行Install.bat安裝服務(wù),成功結(jié)果如下圖:
這時(shí),“我的電腦”右鍵,選擇“管理”,選擇“服務(wù)和應(yīng)用程序”下的“服務(wù)”,就可以看到服務(wù)已安裝,如下圖:
同時(shí),Debug文件夾里有了Logs文件夾,Logs文件夾里有txt文檔,內(nèi)容如下:
可以看到,每分鐘執(zhí)行一次。
三、調(diào)試windows服務(wù)
1)通常的處理辦法是,在service運(yùn)行后, 在調(diào)試器中選擇“附加到進(jìn)程”,附加自己的服務(wù)即可調(diào)試。但此法有局限性,例如在service啟動(dòng)時(shí)的OnStart事件中的代碼, 基本上很難調(diào)試,往往當(dāng)attach到我們的service的時(shí)候,這部分代碼已經(jīng)執(zhí)行過(guò)了。當(dāng)然了,你可以讓OnStart事件之前先睡個(gè)20s,趁著服務(wù)睡覺(jué)的時(shí)候趕緊“附加到進(jìn)程”。System.Threading.Thread.Sleep(1000 * 20);
2)我的做法是,在OnStart事件的最開(kāi)始部分加上“Debugger.Launch();”的調(diào)用, 當(dāng)service運(yùn)行到此處時(shí),將會(huì)彈出一個(gè)選擇調(diào)試器的對(duì)話框,同時(shí)暫停在當(dāng)前位置。這樣,我們就做到了在代碼中手動(dòng)的啟動(dòng)調(diào)試器。
說(shuō)明:a)Debugger.Launch()方法的作用是“啟動(dòng)調(diào)試器并將其連接到進(jìn)程”;
b)可以手動(dòng)設(shè)置斷點(diǎn),也可以用“Debugger.Break();”動(dòng)態(tài)設(shè)置斷點(diǎn);
c)為了避免多個(gè)調(diào)試器實(shí)例,可以用“Debugger.IsAttached”屬性判斷調(diào)試器是否已附加到進(jìn)程,代碼片段:
if (!Debugger.IsAttached) Debugger.Launch();
d)為了使調(diào)試只在Debug模式下生效,Release模式下無(wú)效,可以用條件編譯來(lái)處理,代碼片段如下:
#if DEBUG if (!Debugger.IsAttached) Debugger.Launch(); Debugger.Break(); #endif
關(guān)于條件編譯,請(qǐng)查看我的另一篇博客:C#-#define條件編譯
e)在調(diào)試服務(wù)的其他事件或方法時(shí),同樣可以用到。
彈出選擇調(diào)試器的對(duì)話框,以及調(diào)試界面如下圖所示:
四、卸載windows服務(wù)
卸載服務(wù),同樣以管理員身份運(yùn)行Uninstall.bat即可,成功結(jié)果如下圖:
參考鏈接:https://msdn.microsoft.com/zh-cn/library/windows/desktop/system.diagnostics.debugger(v=vs.110).aspx
源碼下載:http://xiazai.jb51.net/201701/yuanma/WindowsService(jb51.net).rar
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:淺談C#中HttpWebRequest與HttpWebResponse的使用方法
欄 目:C#教程
下一篇:C# Web應(yīng)用調(diào)試開(kāi)啟外部訪問(wèn)步驟解析
本文標(biāo)題:C#使用Windows Service的簡(jiǎn)單教程(創(chuàng)建、安裝、卸載、調(diào)試)
本文地址:http://mengdiqiu.com.cn/a1/C_jiaocheng/6044.html
您可能感興趣的文章
- 01-10C#實(shí)現(xiàn)自定義windows系統(tǒng)日志的方法
- 01-10C#使用Dispose模式實(shí)現(xiàn)手動(dòng)對(duì)資源的釋放
- 01-10C#3.0使用EventLog類寫Windows事件日志的方法
- 01-10C#使用windows服務(wù)開(kāi)啟應(yīng)用程序的方法
- 01-10c# ArrayList的使用方法小總結(jié)
- 01-10C#使用ADO.Net部件來(lái)訪問(wèn)Access數(shù)據(jù)庫(kù)的方法
- 01-10C#使用Mutex簡(jiǎn)單實(shí)現(xiàn)程序單實(shí)例運(yùn)行的方法
- 01-10使用Nopcommerce為商城添加滿XX減XX優(yōu)惠券功能
- 01-10同步調(diào)用和異步調(diào)用WebService
- 01-10C#調(diào)用WebService實(shí)例開(kāi)發(fā)


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹(shù)的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問(wèn)題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-10C#通過(guò)反射獲取當(dāng)前工程中所有窗體并
- 01-10關(guān)于ASP網(wǎng)頁(yè)無(wú)法打開(kāi)的解決方案
- 01-10WinForm限制窗體不能移到屏幕外的方法
- 01-10WinForm繪制圓角的方法
- 01-10C#實(shí)現(xiàn)txt定位指定行完整實(shí)例
- 01-10WinForm實(shí)現(xiàn)仿視頻播放器左下角滾動(dòng)新
- 01-10C#停止線程的方法
- 01-10C#實(shí)現(xiàn)清空回收站的方法
- 01-10C#通過(guò)重寫Panel改變邊框顏色與寬度的
- 01-10C#實(shí)現(xiàn)讀取注冊(cè)表監(jiān)控當(dāng)前操作系統(tǒng)已
隨機(jī)閱讀
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-11Mac OSX 打開(kāi)原生自帶讀寫NTFS功能(圖文
- 04-02jquery與jsp,用jquery
- 01-10SublimeText編譯C開(kāi)發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什