• 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;
            }
        }
    }
  • 相关阅读:
    潜入ICU的四逆汤
    经方医的视角
    黄连解毒汤治疗月经过多
    柳暗花明又一方
    PHP 相关性系数计算
    备忘-VSCODE、apache配置
    c# 基于文件系统实现的队列处理类
    notepad++ 快速运行PHP代码
    dat.gui stats.js 通用参数配置及图像统计工具
    AutoHotkey 自动化脚本工具实例
  • 原文地址:https://www.cnblogs.com/TNSSTAR/p/2654423.html
Copyright © 2020-2023  润新知