我在Windows窗体应用程序(Visual C#)中有一个计时器,当我想退出应用程序时会导致问题。
计时器被定义为表格类的成员:
partial class Form1
{
//These are the members in question:
internal ComACRServerLib.Channel channel;
private System.Timers.Timer updateStuff;
}
在表单应用程序的构造函数中声明/构造计时器:
public Form1()
{
InitializeComponent();
updateStuff = new System.Timers.Timer();
updateStuff.Elapsed += new System.Timers.ElapsedEventHandler(updateStuff_Elapsed);
}
只需按一下按钮即可启动和配置计时器:
private void btnAcquire_Click(object sender, EventArgs e)
{
updateStuff.Interval = 100;
updateStuff.Enabled = true;
updateStuff.AutoReset = true;
updateStuff.Start();
}
当计时器过去时,它会调用updateStuff_Elapsed
,这会获得与setText
一起显示的信息(还有一些代码可以确保调用setText
是线程 - 安全)。
private void updateStuff_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
if (!channel.isOffline)
{
object[] status = channel.GetACRCustom("P6144");
setText(System.Convert.ToString(status[0]));
}
}
public delegate void setTextDelegate(string text);
public void setText(string text)
{
if (this.lblTest.InvokeRequired == true)
{
setTextDelegate d = new setTextDelegate(setText);
this.Invoke(d, new object[] { text });
}
else
{
lblTest.Text = text;
}
}
在应用程序退出时,我尝试摆脱计时器,并使用以下内容阻止它再次触发:
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
updateStuff.AutoReset = false;
updateStuff.Stop();
updateStuff.Close();
updateStuff.Dispose();
components.Dispose();
}
base.Dispose(disposing);
}
但是如果计时器是自动运行的并且我退出程序,我总是得到错误,Timer Elapsed事件updateStuff_elapsed
调用的例程正在尝试使用已经处理过的资源!尽管在处理之前我已经尽力停止并销毁计时器。
如何在应用程序关闭时停止计时器?
修改
我尝试移动Dispose
代码以试图强制定时器关闭但没有运气。在停止和处理之前,我还尝试使用updateStuff.Elapsed -= updateStuff_Elapsed
删除事件调用;
protected override void Dispose(bool disposing)
{
//now this code HAS to run always.
updateStuff.Elapsed -= updateStuff_Elapsed;
updateStuff.AutoReset = false;
updateStuff.Stop();
updateStuff.Close();
updateStuff.Dispose();
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
答案 0 :(得分:2)
正如System.Timers.Timer的文档中所述,Timer的事件处理程序调用在ThreadPool线程上排队。因此,您必须假设事件处理程序可以一次调用多次,或者可以在禁用Timer后调用。因此,事件处理程序必须设计为正确处理这些情况。
首先,将timer的SynchronizingObject属性设置为Form的实例。这将把所有事件处理程序调用编组到UI线程,因此我们不需要打扰表单字段的锁定(我们将始终从同一个UI线程访问所有内容)。使用此属性集,您也不需要在setText方法中调用this.Invoke(...)。
public Form1()
{
updateStuff = new System.Timers.Timer();
updateStuff.SynchronizingObject = this;
...
}
public void setText(string text)
{
lblTest.Text = text;
}
然后创建标志,让你知道,计时器是否被处理掉了。然后只需在事件处理程序中检查此标志:
partial class Form1
{
private bool Disposed;
....
}
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
updateStuff.Dispose();
Disposed = true;
}
base.Dispose(disposing);
}
private void updateStuff_Elapsed(object sender, System.Timers.ElapsedEventArgs e)
{
if(!Disposed)
{
if (!channel.isOffline)
{
object[] status = channel.GetACRCustom("P6144");
setText(System.Convert.ToString(status[0]));
}
}
}
答案 1 :(得分:0)
处理dispose事件为时已晚(在垃圾回收期间)。您需要在FormClosing事件上执行清理过程。
答案 2 :(得分:0)
扩展罗恩和马克所说的话,
解决方案是覆盖OnFormClosing
子类Form
的{{1}}方法:
Form1
另一种解决方案是将计时器放在与用户界面相同的线程上,正如Nuf所说。