0

我有一个非常奇怪的错误,我似乎无法弄清楚。我会尽我所能把这个布置好。

上下文:我有一个应用程序,它会不断地对一组位于不同位置的 IP 进行 ping 操作。(我们需要查看每个设备是否在给定时间启动)

错误:当 ping 主机时,他们的互联网已关闭,应用程序显示所有设备都已关闭(应该如此)。但是当关闭应用程序时,进程(主应用程序 exe)仍在运行,我们的路由器流量显示持续 ping 仍在运行。(相同的时间间隔)。所以它就像计时器一直在走。

但是当应用程序关闭并且组启动时,进程会正常关闭,并且所有 ping 都会停止。

我可以提供你们想看的任何代码。先介绍一下基本情况:

我有一个名为 EquipmentObj 的类。在构造函数内部,启动了一个连续的 ping(异步)*注意,这一切都是在单独的表单上完成的,而不是在主表单上完成的。构造函数和属性:

System.Timers.Timer TimerObj { get; set; }
public EquipmentObj(string ipAddress)
        {
                this.IP_Address = ipAddress;
                this.TimerObj = new System.Timers.Timer();
                this.TimerObj.Interval = 2000;
                this.TimerObj.Elapsed += TimerObj_Elapsed;
                this.TimerObj.AutoReset = true;
                this.start();
        }

开始和经过的方法:

private void start()
        {
            this._requestStop = false;
            this.TimerObj.Start();
        }


        private void TimerObj_Elapsed(object sender, System.Timers.ElapsedEventArgs     e)
            {
                if(!_requestStop)
                    base.start_Ping(this.Ip_Address);
            }

基类:

public void start_Ping(string ipAddress)
        {
            try
            {
                Ping pingSender = new Ping();


                // Create the event Handler 
                pingSender.PingCompleted += pingSender_PingCompleted;

                // Create the buffer
                // Reference buff size: "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
                byte[] packetData = Encoding.ASCII.GetBytes("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa");
                // Create the pingOptions object
                PingOptions packetOptions = new PingOptions();
                // Parse the passed ip address to a proper ipaddress object
                IPAddress ip = System.Net.IPAddress.Parse(ipAddress);

                // Send async
                pingSender.SendAsync(ip, 1050, packetData, ip);
            }
            catch (Exception ex)
            {
                Error_Log.write_log(ex);
            }
        }

        public virtual void pingSender_PingCompleted(object sender, PingCompletedEventArgs e)
        {
            throw new NotImplementedException();
        }

实现 Ping 完成:

  public override void pingSender_PingCompleted(object sender, PingCompletedEventArgs e)
        {
            if (e.Reply.Status == IPStatus.Success)
                this.status = "On Line";
            else
            {
                this.status = "Off Line";
                this.setBounced = ++this.setBounced  ?? 1;
            }
            this.setResponse = e.Reply.RoundtripTime.ToString();
        }

最后我的清理在表单关闭时调用:

 foreach (EquipmentObj obj in this.objectListView1.Objects)
                {
                    obj.Stop();
                }

这里是停靠点:

public void Stop()
       {
           this._requestStop = true; 
           this.TimerObj.Stop();
       }

我试过的:

  • 将清理(上图)添加到结束事件中。
  • 将我的计时器从 system.threading 计时器移至 system.timer 计时器。
  • 我试过实现一次性的。
  • 在表单关闭事件中将 Timers 设置为 null。
  • 在它自己的计时器上调用 dispose。

这些都没有奏效。我希望我没有用太多的代码和写作压倒你们所有人。任何帮助,将不胜感激。

4

1 回答 1

0

My best bet is you have an exception being thrown prior to the Stop being called. I will put some more try { } catch statements in there so you can handle errors.

You may have some threading issues where you are trying to access objects not created on the same thread.

I would check if you need to Invoke anytime you mess with the timer object.

void Stop()
{
    if (objectListView1.InvokeRequired)
    {
         objectListView1.BeginInvoke(new Action(Stop));
         return;
    }

    foreach (EquipmentObj obj in this.objectListView1.Objects)
    {
        obj.Stop();
    }
}
于 2013-03-21T20:29:39.317 回答