Windows service是一種系統自動的、無人值守的程序(僅存在於Windows NT、2000和XP操作系統中),它能夠在系統啓動時開始運行。用戶可以通過Service Control Manager (SCM) applet或者一些特殊的service-control應用來訪問Windows service,使服務在沒有用戶登錄到系統之前執行。
在.NET出現以前,編寫Windows服務是VC++、Delphi才能辦到的事情,
VB必須使用第三方控件纔可以辦到,而且編寫起來特別的複雜。使用Microsoft®.NET Framework,我們可通過創建作爲服務安裝的應用程序來方便地創建Windows服務。
設計:
一個Windows服務程序,按照配置文件中的配置,在指定時刻運行指定程序。
流程:
啓動服務à讀取配置文件à啓動定時器
定時器定時觸發(比如每隔30秒)à循環需要運行組件時間à時間到à運行指定程序
編寫:
創建一個Windows Service
將Server1.cs更名爲SchedulerServer.cs,
雙擊SchedulerServer.cs打開設計頁面,從工具欄的組件中拖Timer控件。
更名爲SchedulerTimer,並設置Enabled爲flase。
注意必須是Components裏面的Timer控件,Windows Forms裏面的Timer控件不行。
F7瀏覽代碼可以看到如下代碼
服務器啓動的時候運行:
/// <summary>
/// Set things in motion so your service can do its work.
/// </summary>
protected override void OnStart(string[] args)
{
// TODO: Add code here to start your service.
}
服務停止的時候運行:
/// <summary>
/// Stop this service.
/// </summary>
protected override void OnStop()
{
// TODO: Add code here to perform any tear-down necessary to stop your service.
}
添加寫日誌函數,用來記錄服務的日誌:
public static void WriteLog(string strLog)
{
string strPath;
strPath=System.Environment.SystemDirectory;
strPath+=@"/SchedulerServer.txt";
FileStream fs = new FileStream(strPath,FileMode.OpenOrCreate,FileAccess.Write);
StreamWriter m_streamWriter = new StreamWriter(fs);
m_streamWriter.BaseStream.Seek(0, SeekOrigin.End);
m_streamWriter.WriteLine(strLog);
m_streamWriter.Flush();
m_streamWriter.Close();
fs.Close();
}
自動服務配置文件
配置文件保存在系統目錄,Windows 2000爲WinNT/System32,文件名爲SchedulerServer.xml,存儲了自動服務的所有配置信息,下面我們來看SchedulerServer.xml的格式:
<SchedulerServer>
<AutoServer>
<FilePath>C:/AutoBackup.dll</FilePath>
<RunTime>02:00</RunTime>
</AutoServer>
<AutoServer>
<FilePath>D:/AutoMail.dll</FilePath>
<RunTime>03:00</RunTime>
</AutoServer>
</SchedulerServer>
FilePath設置需要運行的組件的路徑
RunTime設置需要運行的時間
如果有多個程序需要運行,只需要添加<AutoServer>節點
程序中添加讀取配置文件函數:
private bool ReadConf()
{
try
{
string strPath;
XmlDocument xmldoc=new XmlDocument();
XmlNodeList xmlnd;
strPath=System.Environment.SystemDirectory+@"/SchedulerServer.xml";
xmldoc.Load(strPath);
xmlnd=xmldoc.SelectNodes("SchedulerServer/AutoServer");
arrConf=new String[2,xmlnd.Count];
for(int i=0;i<xmlnd.Count;i++)
{
arrConf[0,i]=xmlnd[i].SelectSingleNode("FilePath").InnerXml.Trim();
arrConf[1,i]=xmlnd[i].SelectSingleNode("RunTime").InnerXml.Trim();
}
return true;
}
catch(Exception e)
{
WriteLog(DateTime.Now.ToString());
WriteLog("Read Configuration Error:");
WriteLog(e.ToString());
return false;
}
}
啓動服務:
定義兩個變量:
private string[,] arrConf;保存配置信息
private Assembly[] assObj;加載組件
在OnStart事件中添加如下代碼:
protected override void OnStart(string[] args)
{
WriteLog("/************************************************************/");
WriteLog("ScheculerServer Start at "+DateTime.Now.ToString());
//Load Configuation
if(!ReadConf())return;
//Load Assembly
try
{
assObj=new Assembly[arrConf.GetLength(1)];
for(int i=0;i<assObj.Length;i++)
{
assObj[i]=Assembly.LoadFrom(arrConf[0,i].ToString());
arrConf[0,i]="NotRuning";
}
}
catch(Exception e)
{
WriteLog(DateTime.Now.ToString());
WriteLog("Load Dll Error:");
WriteLog(e.ToString());
}
//Start Time
SchedulerTimer.Interval=30000;//設置每30秒觸發
SchedulerTimer.Enabled=true;//啓動定時器
}
定時器觸發:
此處完成檢查時間是否運行
private void SchedulerTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
try
{
//SchedulerTimer.Enabled=false;
DateTime dtNow=DateTime.Now;
DateTime dtRun;
for(int i=0;i<arrConf.GetLength(1);i++)
{
dtRun=Convert.ToDateTime(arrConf[1,i].ToString());
if(dtRun.AddSeconds(-30)<=dtNow && dtNow<=dtRun.AddSeconds(30)) {
if(arrConf[0,i].ToString().Trim()=="NotRuning")
{
foreach(Type t in assObj[i].GetTypes())
{
if(t.IsClass && !t.IsAbstract && t.IsPublic)
{
Object obj=Activator.CreateInstance(t);
MethodInfo mi=t.GetMethod("Run");
if(mi!=null)
{
mi.Invoke(obj,null);
obj=null;
GC.Collect();
break;
}
obj=null;
GC.Collect();
}
}
arrConf[0,i]="OnRuning";
WriteLog("-------------------------------------------------------------");
WriteLog(DateTime.Now.ToString()+":Runing "+arrConf[1,i]+" "+assObj[i].Location.ToString());
WriteLog("-------------------------------------------------------------");
}
}
catch(Exception ex)
{
WriteLog("##########################################################");
WriteLog(ex.ToString());
WriteLog("##########################################################");
}
}
這時程序的主骨架就完成了,下一步需要製作Windows服務安裝程序,切換到設計頁面,在屬性的右下腳有Add Installer字樣,單擊,VS.NET自動幫你生成安裝程序。
打開ProjectInstaller.cs文件,可以看到兩個組件,設置其屬性。
運行帳號選擇LocalSystem,系統以本地系統帳號運行。
啓動類型選擇自動。
編譯爲exe,打開VS.NET cmd,進入exe目錄,運行InstallUtil.exe SchedulerServer.exe安裝服務,成功之後,打開服務管理可以看到剛纔的服務,啓動服務即可。
服務日誌保存在系統目錄下的SchedulerServer.txt
/************************************************************/
ScheculerServer Start at 12/24/2003 3:46:21 PM
-------------------------------------------------------------
12/24/2003 3:46:51 PM:Runing 15:47 e:/work/geid/src/geidautocheck/bin/debug/geidautocheck.dll
-------------------------------------------------------------
ScheculerServer Stop at 12/24/2003 3:48:55 PM
/************************************************************/
自動運行程序:
1..NET組件(dll)
2.入口爲
public void Run()
{}
啓動服務:
定義兩個變量:
private string[,] arrConf;保存配置信息
private Assembly[] assObj;加載組件
在OnStart事件中添加如下代碼:
protected override void OnStart(string[] args)
{
WriteLog("/************************************************************/");
WriteLog("ScheculerServer Start at "+DateTime.Now.ToString());
//Load Configuation
if(!ReadConf())return;
//Load Assembly
try
{
assObj=new Assembly[arrConf.GetLength(1)];
for(int i=0;i<assObj.Length;i++)
{
assObj[i]=Assembly.LoadFrom(arrConf[0,i].ToString());
arrConf[0,i]="NotRuning";
}
}
catch(Exception e)
{
WriteLog(DateTime.Now.ToString());
WriteLog("Load Dll Error:");
WriteLog(e.ToString());
}
//Start Time
SchedulerTimer.Interval=30000;//設置每30秒觸發
SchedulerTimer.Enabled=true;//啓動定時器
}
定時器觸發:
此處完成檢查時間是否運行
private void SchedulerTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
try
{
//SchedulerTimer.Enabled=false;
DateTime dtNow=DateTime.Now;
DateTime dtRun;
for(int i=0;i<arrConf.GetLength(1);i++)
{
dtRun=Convert.ToDateTime(arrConf[1,i].ToString());
if(dtRun.AddSeconds(-30)<=dtNow && dtNow<=dtRun.AddSeconds(30)) {
if(arrConf[0,i].ToString().Trim()=="NotRuning")
{
foreach(Type t in assObj[i].GetTypes())
{
if(t.IsClass && !t.IsAbstract && t.IsPublic)
{
Object obj=Activator.CreateInstance(t);
MethodInfo mi=t.GetMethod("Run");
if(mi!=null)
{
mi.Invoke(obj,null);
obj=null;
GC.Collect();
break;
}
obj=null;
GC.Collect();
}
}
arrConf[0,i]="OnRuning";
WriteLog("-------------------------------------------------------------");
WriteLog(DateTime.Now.ToString()+":Runing "+arrConf[1,i]+" "+assObj[i].Location.ToString());
WriteLog("-------------------------------------------------------------");
}
}
catch(Exception ex)
{
WriteLog("##########################################################");
WriteLog(ex.ToString());
WriteLog("##########################################################");
}
}
這時程序的主骨架就完成了,下一步需要製作Windows服務安裝程序,切換到設計頁面,在屬性的右下腳有Add Installer字樣,單擊,VS.NET自動幫你生成安裝程序。
打開ProjectInstaller.cs文件,可以看到兩個組件,設置其屬性。
運行帳號選擇LocalSystem,系統以本地系統帳號運行。
啓動類型選擇自動。
編譯爲exe,打開VS.NET cmd,進入exe目錄,運行InstallUtil.exe SchedulerServer.exe安裝服務,成功之後,打開服務管理可以看到剛纔的服務,啓動服務即可。
服務日誌保存在系統目錄下的SchedulerServer.txt
/************************************************************/
ScheculerServer Start at 12/24/2003 3:46:21 PM
-------------------------------------------------------------
12/24/2003 3:46:51 PM:Runing 15:47 e:/work/geid/src/geidautocheck/bin/debug/geidautocheck.dll
-------------------------------------------------------------
ScheculerServer Stop at 12/24/2003 3:48:55 PM
/************************************************************/
自動運行程序:
1..NET組件(dll)
2.入口爲
public void Run()
{}