C# 实现守护进程

这段时间在做一个关于数据交互的项目。接收到客户发送过来的文件后,通过Windows服务将文件按一定的规则分发到不同的MQ消息队列,然后再由不同的处理程序处理。虽然在编码中尽可能的考虑到了异常以及记录了详细的日志,但是服务还是偶尔抽风停掉了,这样就造成了文件堆积,客户请求得不到及时的响应。所以需要一个守护进程来保证服务始终是启动状态的。

首先,要保证需要监控的进程可配置,以及指定日志的保存位置。在App.config中配置日志保存路径以及需监控进程和对应的exe可执行程序的位置(之前考虑是如果没有该服务,则使用exe安装服务)。

  <appSettings>
    <add key="LogPath" value="D:\Study"/>
    <add key="YTGDataExchangeProcess" value="d:\Study\YTGDataExchangeProcess.exe"/>
  </appSettings>
public class ServiceSearch
    {
        static List<ServiceProcessInfo> _ServiceList = new List<ServiceProcessInfo>();
        static object _logLocker = new Object();
        //日志记录位置 
        static readonly string _logPath;

        static ServiceSearch()
        {
            try
            {
                string[] appSettingsKeys = ConfigurationManager.AppSettings.AllKeys;

                foreach (string item in appSettingsKeys)
                {
                    string value = ConfigurationManager.AppSettings[item].Trim();
                    //日志配置
                    if (item == "LogPath")
                    {
                        if (!string.IsNullOrEmpty(value))
                        {
                            _logPath = string.Format("{0}\\ProcessMonitorLog", value);
                            if (!Directory.Exists(_logPath))
                            {
                                Directory.CreateDirectory(_logPath);
                            }
                        }
                    }
                    else
                    {
                        if (!string.IsNullOrEmpty(value))
                        {
                            //应用程序是否存在 
                            if (File.Exists(value))
                            {
                                _ServiceList.Add(new ServiceProcessInfo
                                {
                                    ServiceExePath = value,
                                    ServiceName = item
                                });
                            }
                        }

                    }

                }
            }
            catch (Exception ex)
            {
                WriteLog(ex.Message);
            }
        }

        public static void StartMonitor()
        {
            if (_ServiceList.Count != 0)
            {
                foreach (ServiceProcessInfo info in _ServiceList)
                {
                    string sevName = info.ServiceName;
                    string sevExePath = info.ServiceExePath;

                    if (!string.IsNullOrEmpty(sevExePath))
                    {
                        //校验exe文件是否存在
                        if (File.Exists(sevExePath))
                        {
                            ScanServiceManager(sevName, sevExePath);
                        }
                    }
                }
            }
            else
            {
                WriteLog("没有配置需要监控的服务!");
            }
        }

        /// <summary>
        /// 查找服务的进程
        /// 有则判断进程是否运行,非运行状态则启动
        /// </summary>
        public static void ScanServiceManager(string serviceName, string serviceExePath)
        {
            try
            {
                ServiceController sevCtrler = new ServiceController(serviceName);
                //该服务已存在
                //如果服务状态不为启动
                if (sevCtrler.Status != ServiceControllerStatus.Running)
                {
                    sevCtrler.Start();
                }
                //创建监控线程
                WatchService(serviceName);
            }
            catch (Exception ex)
            {
                //该服务不存在
                WriteLog(string.Format(ex.Message));
            }
        }

        /// <summary>
        /// 为配置的进程添加监控进程
        /// </summary>
        /// <param name="pro"></param>
        /// <param name="processAddress"></param>
        public static void WatchService(string ServiceName)
        {
            ServiceMonitor monitor = new ServiceMonitor(ServiceName);
            Thread thread = new Thread(new ThreadStart(monitor.Monitor));
            thread.IsBackground = true;
            thread.Start();
        }

        /// <summary>
        /// 写日志 
        /// </summary>
        /// <param name="errMessage"></param>
        public static void WriteLog(string errMessage)
        {
            string logPath = _logPath;
            //没有配置日志目录,不记录
            if (!string.IsNullOrEmpty(logPath))
            {
                lock (_logLocker)
                {
                    string fullName = string.Format("{0}\\{1}.log", logPath, DateTime.Now.ToString("yyyy-MM-dd"));
                    if (!File.Exists(fullName))
                    {
                        File.Create(fullName).Close();
                    }
                    using (StreamWriter sw = new StreamWriter(fullName, true, Encoding.UTF8))
                    {
                        sw.WriteLine(String.Format("[{0}]{1}", DateTime.Now.ToString("hh:mm:ss fff"), errMessage));
                        sw.Close();
                    }
                }
            }
        }

        public class ServiceProcessInfo
        {
            /// <summary>
            /// 服务名称
            /// </summary>
            public string ServiceName
            {
                get;
                set;
            }
            /// <summary>
            /// 应用程序位置
            /// </summary>
            public string ServiceExePath
            {
                get;
                set;
            }
        }


    }

上面ScanServiceManager方法中,使用serviceName实例化ServiceController对象,并启用一个单独的线程对其轮询,监控其状态;否则只会对第一个服务进行监控后阻塞。

    public class ServiceMonitor
    {
        /// <summary>
        /// 服务名称
        /// </summary>
        private string _ServiceName;

        public string ServiceName
        {
            get { return _ServiceName; }
            set { _ServiceName = value; }
        }

        public ServiceMonitor(string ServiceName)
        {
            this._ServiceName = ServiceName;
        }

        /// <summary>
        /// 监听服务
        /// </summary>
        public void Monitor()
        {
            while (true)
            {
                try
                {
                    ServiceController ctrler = new ServiceController(_ServiceName);
                    if (ctrler.Status != ServiceControllerStatus.Running)
                    {
                        ServiceSearch.WriteLog(string.Format("正在启动服务{0}...", _ServiceName));
                        ctrler.Start();
                        ServiceSearch.WriteLog(string.Format("服务{0}启动成功!", _ServiceName));
                    }
                }
                catch (Exception ex)
                {
                    ServiceSearch.WriteLog(string.Format("服务{0}启动失败,错误原因:{1}", _ServiceName, ex.Message));
                }
                Thread.Sleep(1000 * 40);
            }
        }

    }

 

posted @ 2015-08-03 22:37  DeqiangLee  阅读(6400)  评论(0编辑  收藏  举报