C# windows 服务 创建调试

前言:编写一个Windows服务程序,定时从数据库中拿出记录发送邮件。测试环境:Visual Studio 2005 SP1、Windows Server 2003 SP2

一、新建项目

打开VS2005,新建一个“Windows 服务”项目。

二、添加Timer

展开“工具箱”,在“组件”标签下找到“Timer”双击,这时就添加了一个Timer组件,修改“Name”属性为“timEmail”、“Enabled”为“false”、“Interval”为“60000”。

接下来要做一些修补工作,不知是VS2005的BUG还是我没找着地方,在VS2003下是不存在该问题的:刚从“组件”下添加的“Timer”按理说应该来自“System.Timers命名空间”,也只有“System.Timers.Timer”才能在Windows服务程序中正常工作,但是现在这个Timer却是属于“System.Windows.Forms.Timer”的。所以得稍作修改,打开“.Designer.cs”文件,修改如下:

#region 组件设计器生成的代码
//........以上略
/// <summary> 
/// 设计器支持所需的方法 - 不要
/// 使用代码编辑器修改此方法的内容。
/// </summary>
private void InitializeComponent()
{
            this.components = new System.ComponentModel.Container();
            //this.timEmail = new System.Windows.Forms.Timer(this.components);原
           this.timEmail = new System.Timers.Timer();//
            this.timEmail.Interval = 60000;
            this.ServiceName = "Service1";
}
#endregion
//private System.Windows.Forms.Timer timEmail;原
private System.Timers.Timer timEmail;//改三、添加配置文件

服务每次调用配置文件,获取一些基本参数,这样一些变更就可直接修改配置文件而不必修改代码。新建ServiceConfig.xml存放于项目“Bin\Debug\”下:

<?xml version="1.0" encoding="utf-8" ?> 
<serviceConfig>
    <serviceItem 
        name="sendEmail" 
         enable="true" 
         elapsed="60000" 
         connectionString="your database connection..." 
         smtp="smtp address" 
         account="your email account..." 
         password="your password..." >
    </serviceItem>
</serviceConfig>四、以下是实现代码

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.ServiceProcess;
using System.Text;
using System.Xml;//操作配置文件
using System.IO;//写日志
using System.Threading;//使用线程

namespace ClientWindowsService
{
    public partial class ClientService : ServiceBase
     {
        public ClientService()
         {
             InitializeComponent();
         }

        protected override void OnStart(string[] args)
         {
            //服务启动
      this.timEmail.Enabled = true;
            this.tSendEmail();
         }

        protected override void OnStop()
         {
            //服务停止
      this.timEmail.Enabled = false;
         }

        private void timEmail_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
         {
            //定时器
      this.tSendEmail();
         }

        //开启新进程发送邮件
    private void tSendEmail()
         {
             Thread t = new Thread(new ThreadStart(sendEmail));
             t.Start();
         }

        //发送邮件函数
    private void sendEmail()
         {
             XmlDocument doc = new XmlDocument();
            //添加System.Windows.Forms引用,获取执行目录
      string configFile = System.Windows.Forms.Application.StartupPath.ToString() + "\ServiceConfig.xml";
             doc.Load(@configFile);
             XmlElement root = doc.DocumentElement;
            foreach (XmlNode node in root)
             {
                //如果配置文件中开启服务
        if (node.Attributes["name"].Value == "sendEmail" && node.Attributes["enable"].Value == "true")
                 {
                    try
                     {
                        //读取数据库,发送邮件操作,略
                     }
                    catch (Exception error)
                     {
                        //写错误日志
            using (StreamWriter sw = new StreamWriter(System.Windows.Forms.Application.StartupPath.ToString() + @"" + DateTime.Now.ToString("yyyy-MM-dd") + ".txt", true, System.Text.Encoding.UTF8))
                         {
                             sw.WriteLine(DateTime.Now.ToString() + "");
                             sw.WriteLine(error.ToString());
                             sw.WriteLine("---------------------------------------------");
                             sw.Close();
                         }
                     }
                 }
             }//end foreach
         }

     }//end class
}//end namespace五、布署服务

在设计模式下右键-->添加安装程序-->设置serviceProcessInstaller1的Account为LocalSystem

设置serviceInstaller1的StartType为Automatic

编译

在命令模式下执行:%systemroot%\microsoft.net\framework\v2.0.50727\installUtil.exe D:\项目目录\bin\Debug\可执行文件名.exe
在每次需要修改Windows服务时,这就会要求你卸载和重新安装这个服务。不过要注意在卸载这个服务前,最好确保服务管理控制台已经关闭,这会是一个很好的习惯。如果没有这样操作的话,你可能在卸载和重安装Windows服务时会遇到麻烦。仅卸载服务的话,可以执行相的InstallUtil命令用于注销服务,不过要在后面加一个/u命令开关。

调试Windows服务

  从另外的角度度看,调试Windows服务绝不同于一个普通的应用程序。调试Windows服务要求的步骤更多。服务不能象你对普通应用程序做的那样,只要简单地在开发环境下执行就可以调试了。服务必须首先被安装和启动,这一点在前面部分我们已经做到了。为了便于跟踪调试代码,一旦服务被启动,你就要用Visual Studio把运行的进程附加进来(attach)。记住,对你的Windows服务做的任何修改都要对这个服务进行卸载和重安装。


附加正在运行的Windows服务
  为了调试程序,有些附加Windows服务的操作说明。这些操作假定你已经安装了这个Windows服务并且它正在运行。

1. 用Visual Studio装载这个项目 
2. 点击“调试”菜单
3. 点击“进程”菜单
4. 确保 显示系统进程 被选
5. 在 可用进程 列表中,把进程定位于你的可执行文件名称上点击选中它
6. 点击 附加 按钮
7. 点击 确定
8. 点击 关闭
9. 在timer1_Elapsed方法里设置一个断点,然后等它执行
 

自己实现完成的:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.ServiceProcess;
using System.Text;
using System.IO;
using System.Windows.Forms;
using System.Threading;
using System.Xml;

namespace MyNT
{
    public partial class Service1 : ServiceBase
    {
        public Service1()
        {
            InitializeComponent();
        }

        protected override void OnStart(string[] args)
        {
            this.timEmail.Enabled = true;
            tStart();
        }
        protected override void OnStop()
        {
            tStop();
            this.timEmail.Enabled = false;
        }
        private void tStart()
        {
            Thread t = new Thread(new ThreadStart(Start1));
            t.Start();
        }
        private void tStop()
        {
            Thread t = new Thread(new ThreadStart(Stop1));
            t.Start();
        }
        private void timEmail_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
        {
            //定时器
            Start1();
        }
        private void Start1()
        {
            FileStream fs = new FileStream(@"d:\mcWindowsService.txt", FileMode.OpenOrCreate, FileAccess.Write);
            StreamWriter m_streamWriter = new StreamWriter(fs);
            m_streamWriter.BaseStream.Seek(0, SeekOrigin.End);
            m_streamWriter.WriteLine("MyLogServer: 服务开始" + DateTime.Now.ToString() + "\n");
            m_streamWriter.Flush();
            m_streamWriter.Close();
            fs.Close();
        }
        private void Stop1()
        {
            FileStream fs = new FileStream(@"d:\mcWindowsService.txt", FileMode.OpenOrCreate, FileAccess.Write);
            StreamWriter m_streamWriter = new StreamWriter(fs);
            m_streamWriter.BaseStream.Seek(0, SeekOrigin.End);
            m_streamWriter.WriteLine(" MyLogServer: 服务结束 " + DateTime.Now.ToString() + "\n");
            m_streamWriter.Flush();
            m_streamWriter.Close();
            fs.Close();
        }
        /// <summary>
        /// XML的相应信息获取操作XML
        /// </summary>
        /// <returns></returns>
        public static string GetThreeLevelGroups()
        {
            string returnValue = "";
            XmlDocument XMlDoc = new XmlDocument();
            XMlDoc.Load(System.Web.HttpContext.Current.Server.MapPath("/" ));
            XmlNodeList nodelist = XMlDoc.SelectSingleNode("GroupInfo").ChildNodes;

            foreach (XmlNode xn_1 in nodelist)
            {
                XmlElement element1 = (XmlElement)xn_1;
                XmlNodeList nls_2 = element1.ChildNodes;

                foreach (XmlNode xn_2 in nls_2)
                {
                    XmlElement element2 = (XmlElement)xn_2;
                    XmlNodeList nls_3 = element2.ChildNodes;
                    foreach (XmlNode xn_3 in nls_3)
                    {
                        XmlElement element3 = (XmlElement)xn_3;
                        returnValue += "," + element3.GetAttribute("GroupName");
                    }
                }
            }
            return returnValue;
        }
    }
}

 

posted @ 2012-08-24 16:25  Star★  阅读(1181)  评论(0编辑  收藏  举报