我试图通过使用计时器延迟我的方法中的事件,但是我不一定了解如何使用计时器等待。
我将我的计时器设置为2秒,但是当我运行此代码时,最后一次调用运行时没有2秒延迟。
Timer timer = new Timer();
timer.Tick += new EventHandler(timer_Tick); // Everytime timer ticks, timer_Tick will be called
timer.Interval = (1000) * (2); // Timer will tick evert second
timer.Enabled = true; // Enable the timer
void timer_Tick(object sender, EventArgs e)
{
timer.Stop();
}
private void button1_Click(object sender, EventArgs e)
{
label1.Text = "first";
timer.Start();
label1.Text = "second";
}
所以当我点击我的按钮时,它会立即将label1显示为“second”,而不是更改为“first”,等待2秒,然后更改为“second”。我在这里阅读了许多关于使用计时器而不是thread.sleep的线程,但我似乎无法找到/弄清楚如何实际实现它。
答案 0 :(得分:11)
timer.Start()
只是启动计时器,但在计时器在后台运行时立即返回。因此,在将标签文本设置为first
和second
之间几乎没有停顿。你想要做的是等待定时器勾选,然后再次更新标签:
void timer_Tick(object sender, EventArgs e)
{
timer.Stop();
label1.Text = "second";
}
private void button1_Click(object sender, EventArgs e)
{
label1.Text = "first";
timer.Start();
}
顺便说一下。你不应该将timer.Enabled
设置为true,你已经使用timer.Start()
启动了计时器。
如评论中所述,您可以将计时器创建放入一个方法,如下所示(注意:这是未经测试的):
public void Delayed(int delay, Action action)
{
Timer timer = new Timer();
timer.Interval = delay;
timer.Tick += (s, e) => {
action();
timer.Stop();
};
timer.Start();
}
然后你可以像这样使用它:
private void button1_Click(object sender, EventArgs e)
{
label1.Text = "first";
Delayed(2000, () => label1.Text = "second");
}
使用Delayed是否包含内存泄漏(引用泄漏)?
订阅活动始终会创建双向参考。
在这种情况下,
的循环引用timer.Tick
获取对匿名函数(lambda)的引用。该函数提升局部变量timer
,虽然它是引用而不是值,并且包含对传入的Action委托的引用。该委托将包含label1
的引用,Form
的实例成员。那么是否存在从Timer
到Form
?我不知道答案,我发现它有点难以推理。因为我不知道,除了停止定时器(这是方法的
Delayed
参数之外,我会删除sender
中lambda的使用,使其成为正确的方法并拥有它。 ),也删除事件。
通常lambdas不会导致垃圾回收问题。在这种情况下,计时器实例仅存在于本地,并且lambda中的引用不会阻止垃圾收集来收集实例(另请参阅this question)。
我实际上是使用.NET Memory Profiler再次测试了这个。收集的计时器对象很好,没有发生泄漏。该分析器确实给了我一个警告,即有些情况“虽然没有妥善处理,但已经收集了垃圾”。删除事件处理程序本身(通过保持对它的引用)并没有解决这个问题。将捕获的计时器参考更改为(Timer)s
也没有改变。
有什么帮助 - 显然是在停止计时器后在事件处理程序中调用timer.Dispose()
,但我认为这是否真的有必要。我不认为剖析器警告/注释是关键的。
答案 1 :(得分:8)
如果你正在使用C#5.0 await
,那么更多更容易:
private async void button1_Click(object sender, EventArgs e)
{
label1.Text = "first";
await Task.Delay(2000);
label1.Text = "second";
}
答案 2 :(得分:0)
如果你想要做的就是在计时器打勾时更改文字,那么你最好放一下......
label1.Text = "second";
...在计时器勾选中,在将计时器更改为enabled = false之前或之后;
像这样;
void timer_Tick(object sender, EventArgs e)
{
timer.Stop();
label1.Text = "second";
}
private void button1_Click(object sender, EventArgs e)
{
label1.Text = "first";
timer.Start();
}
答案 3 :(得分:0)
private bool Delay(int millisecond)
{
Stopwatch sw = new Stopwatch();
sw.Start();
bool flag = false;
while (!flag)
{
if (sw.ElapsedMilliseconds > millisecond)
{
flag = true;
}
}
sw.Stop();
return true;
}
bool del = Delay(1000);