4

我开发了一个 32 位服务,我在 Windows 7 Home Premium x64 中运行它。问题是当我启动它时,windows 给了我以下消息

本地计算机上的 WLConsumer 服务启动然后停止。如果某些服务没有被其他服务或程序使用,它们会自动停止。

我在事件日志中发现了以下消息

无法启动服务。System.ArgumentException:日志 WLConsumer 已在本地计算机上注册为源。在 System.Diagnostics.EventLogInternal.CreateEventSource(EventSourceCreationData sourceData) 在 System.Diagnostics.EventLogInternal.VerifyAndCreateSource(String sourceName, String currentMachineName) 在 System.Diagnostics.EventLogInternal.WriteEntry(String message, EventLogEntryType type, Int32 eventID, Int16 category, Byte[ ] rawData) 在 System.Diagnostics.EventLog.WriteEntry(String message, EventLogEntryType type) 在 WeblogicConsumerService.WeblogicConsumer.winEventlogMe(String logTxt, String logSrc, Char entryType) 在 C:\Program Files (x86)\CSI\WeblogicConsumerService\WeblogicConsumer。 cs:C 中 WeblogicConsumerService.WeblogicConsumer.OnStart(String[] args) 的第 136 行:

这是我在 OnStart() 方法中的代码块

   protected override void OnStart(string[] args)
    {
        #region WEBLOGIC CREDENTIALS
        try
        {
            //Weblogic URL
            this.url = Registry.LocalMachine.OpenSubKey(@"Software\CSI_WL").GetValue("URL").ToString();

            //Queue name
            this.qName = Registry.LocalMachine.OpenSubKey(@"Software\CSI_WL").GetValue("Queue").ToString();

            //Weblogic login name
            this.user = Registry.LocalMachine.OpenSubKey(@"Software\CSI_WL").GetValue("User").ToString();

            //Weblogic password
            this.pwd = Registry.LocalMachine.OpenSubKey(@"Software\CSI_WL").GetValue("Pwd").ToString();

            //Weblogic Connection Factory
            this.cfName = Registry.LocalMachine.OpenSubKey(@"Software\CSI_WL").GetValue("ConnectionFactory").ToString();

            //root folder
            this.rFolder = Registry.LocalMachine.OpenSubKey(@"Software\CSI_WL").GetValue("root").ToString();
        }
        catch (Exception e)
        {
            winEventlogMe(e.Message, "WLRegistryKeys", 'e');
        }
        #endregion

        winEventlogMe("Successful start", "SeriviceStartup", 'i');
        synchro.Enabled = true;
    }

winEventLogMe 是我调用的日志记录方法。

        public static void winEventlogMe(string logTxt, string logSrc, char entryType)
    {
        #region Log
        //Log to event log

        EventLog theEvent = new EventLog("WLConsumer");
        theEvent.Source = logSrc;
        if (entryType == 'e')
            theEvent.WriteEntry(logTxt, EventLogEntryType.Error);
        else if (entryType == 'i')
            theEvent.WriteEntry(logTxt, EventLogEntryType.Information);
        else if (entryType == 'w')
            theEvent.WriteEntry(logTxt, EventLogEntryType.Warning);
        else
            theEvent.WriteEntry(logTxt, EventLogEntryType.Error);*/
        #endregion
    }

当我在 OnStart() 方法中注释掉对 winEventLogMe() 方法的调用时,服务启动时不会出现错误。所以显然 winEventLogMe() 方法有问题。有人可以帮我找出问题所在,因为我现在完全不知道如何解决这个问题。

提前谢谢:)


@nick_w 我已按照您的建议编辑了我的代码,服务已成功启动。但是在停止它时,我收到以下消息:

未能停止服务。System.ArgumentException:源“WLConsumer2012”未在日志“ServiceStop”中注册。(注册在日志'SeriviceStartup'中。) " Source 和 Log 属性必须匹配,或者您可以将 Log 设置为空字符串,它会自动匹配 Source 属性。在 System.Diagnostics.EventLogInternal.VerifyAndCreateSource (String sourceName, String currentMachineName) 在 System.Diagnostics.EventLogInternal.WriteEntry(String message, EventLogEntryType type, Int32 eventID, Int16 category, Byte[] rawData) 在 System.Diagnostics.EventLog.WriteEntry(String message, EventLogEntryType type) 在 WeblogicConsumerService C中的.WeblogicConsumer.winEventlogMe(String logTxt, String logSrc, Char entryType):

这是 OnStop() 方法

        protected override void OnStop()
    {
        winEventlogMe("Successful stop", "ServiceStop", 'i');
    }

这些事件日志开始让我很困惑。我做过同样的方法登录其他服务,从来没有遇到过这样的问题。我怎么能在这个服务中得到这些错误,但它与我所做的所有其他错误并没有太大区别:(

4

2 回答 2

4

我认为这是你的问题:

EventLog theEvent = new EventLog("WLConsumer");

从例外情况来看,我认为这WLConsumer是事件源的名称。这意味着您可能会更好:

EventLog theEvent = new EventLog(logSrc);
theEvent.Source = "WLConsumer";

这只是反过来使用参数。

如果我做一点反编译,会有这样的检查:

if (!EventLogInternal.SourceExists(logName, machineName, true))

在您的情况下,我认为此检查返回 true,这意味着它正在尝试创建一个名为WLConsumer但失败的日志,因为WLConsumer已注册为事件源。

编辑:

当我过去使用事件日志时,我将所有内容都写入源和日志的相同组合。在您的情况下,您似乎每次编写条目时都使用不同的源和日志组合。

来自MSDN(重点是我的):

如果您写入事件日志,则必须指定或创建事件源。您必须拥有计算机的管理权限才能创建新的事件源。源将您的应用程序与事件日志注册为有效的条目源。您一次只能使用 Source 写入一个日志。源可以是任何随机字符串,但名称必须与计算机上的其他源不同。源通常是应用程序的名称或另一个标识字符串。尝试创建重复的 Source 值会引发异常。但是,单个事件日志可以与多个源相关联。

我的建议是这样的:

使用WLConsumer(或WLConsumer2012)作为您的来源,或者

  1. 定义你自己的日志,“WLConsumerServiceEventLog”或其他东西;或者
  2. 将日志留空。在这种情况下,它们会进入应用程序日志。

无论如何,标准做法似乎是在第一次运行您的服务之前做这样的事情,例如在安装程序中(直接从上面的链接复制):

// Create the source, if it does not already exist. 
if(!EventLog.SourceExists("MySource"))
{
    //An event log source should not be created and immediately used. 
    //There is a latency time to enable the source, it should be created 
    //prior to executing the application that uses the source. 
    //Execute this sample a second time to use the new source.
    EventLog.CreateEventSource("MySource", "MyNewLog");
    Console.WriteLine("CreatedEventSource");
    Console.WriteLine("Exiting, execute the application a second time to use the source.");
    // The source is created.  Exit the application to allow it to be registered. 
    return;
}

请注意评论中关于延迟的点。日志不一定会立即创建,因此在编写代码时要牢记这一点。您还可以使用EventLogInstaller创建日志。如果您使用安装程序来部署服务,这可能是更简单的选择。

于 2012-11-23T01:16:32.620 回答
0

关键是不要重载 on start 方法,为了防止服务启动失败,通常 onstart 方法将主代码作为单独的线程启动

于 2012-11-23T03:02:38.447 回答