入门示例: 假如有一个label,我们希望像走马灯一样,从1显示到100
private void button1_Click(object sender, EventArgs e) { for (int i = 0; i <= 100; i++) { label1.Text = i.ToString(); } }
但这种写法,只会卡一下,然后label显示100,没有变化的. 如果用doEvents,就可以变化了
private void button1_Click(object sender, EventArgs e) { for (int i = 0; i <= 100; i++) { label1.Text = i.ToString(); Thread.Sleep(50); Application.DoEvents(); } }
但是stackoverflow里说Application.DoEvents是为了适应vb项目迁移而设的,尽量不推荐使用
改成多线程,子线程是不能直接修改UI上的控件的,只能通过委托,让UI线程来修改.
delegate void D(object obj); private void button2_Click(object sender, EventArgs e) { for (int i = 0; i < 10; i++) { Thread t = new Thread(new ParameterizedThreadStart(SetTextBoxValue)); t.IsBackground = true; t.Start(i.ToString()); Debug.WriteLine("Thread #"+ i +" Start"); } } void SetTextBoxValue(object obj) { if (label1.InvokeRequired) { Debug.WriteLine("Thread #" + obj + "InvokeRequired,set Delegate"); D d = new D(SetTextBoxValue); label1.Invoke(d, obj); //异步的话,不保证哪个线程先完成.最后的结果是8或9 } else { Debug.WriteLine("Thread #" + obj + " UI Change ");
//An unhandled exception of type 'System.InvalidOperationException' occurred in System.Windows.Forms.dll //Additional information: 线程间操作无效: 从不是创建控件“label1”的线程访问它。 this.label1.Text = obj.ToString(); Thread.Sleep(50); } }
debug的输出如下: 启动10个线程.10个线程把修改控件属性的操作交给委托, 最后再UpdateUI
Thread #0 Start Thread #0InvokeRequired,set Delegate The thread 0x19ec has exited with code 259 (0x103). Thread #1 Start Thread #1InvokeRequired,set Delegate The thread 0x570 has exited with code 259 (0x103). Thread #2 Start Thread #2InvokeRequired,set Delegate The thread 0x1d04 has exited with code 259 (0x103). Thread #3 Start Thread #4 Start Thread #4InvokeRequired,set Delegate The thread 0x1a8c has exited with code 259 (0x103). Thread #5InvokeRequired,set Delegate Thread #3InvokeRequired,set Delegate The thread 0x1a88 has exited with code 259 (0x103). The thread 0xedc has exited with code 259 (0x103). Thread #5 Start Thread #6 Start Thread #7 Start Thread #7InvokeRequired,set Delegate Thread #6InvokeRequired,set Delegate The thread 0x1b58 has exited with code 259 (0x103). The thread 0x694 has exited with code 259 (0x103). Thread #8InvokeRequired,set Delegate The thread 0x1680 has exited with code 259 (0x103). Thread #8 Start Thread #9 Start Thread #0 UI Change Thread #9InvokeRequired,set Delegate The thread 0x177c has exited with code 259 (0x103). Thread #1 UI Change Thread #2 UI Change Thread #4 UI Change Thread #5 UI Change Thread #3 UI Change Thread #7 UI Change Thread #6 UI Change Thread #8 UI Change Thread #9 UI Change The thread 0x67c has exited with code 0 (0x0).