.NET自動服務程序—C#

用程序開發中,常常需要實現這樣一種功能:讓服務器在每天的特定時刻運行固定的程序(或者實現固定的操作),比如讓系統在每天的200備份數據庫數據。要實現這樣的功能,我們可以使用Windows服務(Windows service)。

Windows service是一種系統自動的、無人值守的程序(僅存在於Windows NT2000XP操作系統中),它能夠在系統啓動時開始運行。用戶可以通過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,並設置Enabledflase

注意必須是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 2000WinNT/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()

{}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章