[点晴永久免费OA]C#多线程解决界面卡死问题的完美解决方案
问题描述:
当我们的界面需要在程序运行中不断更新数据时,当一个textbox的数据需要变化时。
为了让程序执行中不出现界面卡死的现像,最好的方法就是多线程来解决,一个主线程来创建界面,使用一个子线程来执行程序并更新主界面,这样就不会出现卡死的现像了。
但是为什么在使用的过程中一样会有很多地方会出现卡死呢?这个问题其实也困或了我很久,但是今天终于解决了,而且我发现很多人有这样的问题,所以我分享一个例子方便大家参考。
先来看看我的界面
当我单击开始执行后
是数据在不断的更新
这个时候界面是不会卡死的,只是数据在不断的更新
下面看看我的代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 | using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Windows.Forms; using System.Threading; namespace WindowsFormsApplication3 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } //创建一个委托,是为访问TextBox控件服务的。 public delegate void UpdateTxt( string msg); //定义一个委托变量 public UpdateTxt updateTxt; //修改TextBox值的方法。 public void UpdateTxtMethod( string msg) { richTextBox1.AppendText(msg + "\r\n" ); richTextBox1.ScrollToCaret(); } //此为在非创建线程中的调用方法,其实是使用TextBox的Invoke方法。 public void ThreadMethodTxt( int n) { this .BeginInvoke(updateTxt, "线程开始执行,执行" + n + "次,每一秒执行一次" ); for ( int i = 0; i < n; i++) { this .BeginInvoke(updateTxt, i.ToString()); //一秒 执行一次 Thread.Sleep(1000); } this .BeginInvoke(updateTxt, "线程结束" ); } //开启线程 private void button1_Click( object sender, EventArgs e) { Thread objThread = new Thread( new ThreadStart( delegate { ThreadMethodTxt(Convert.ToInt32(textBox1.Text.Trim())); })); objThread.Start(); } private void Form1_Load_1( object sender, EventArgs e) { //实例化委托 updateTxt = new UpdateTxt(UpdateTxtMethod); } } } |
上面是全部代码方便大家参考吧
第一步我们先来定义一个委托updateTxt
1 2 3 4 | //创建一个委托,是为访问TextBox控件服务的。 public delegate void UpdateTxt( string msg); //定义一个委托变量 public UpdateTxt updateTxt; |
主要是使用一个委托来更新界面的richTextBox1
实例方法如下
1 2 3 4 5 | private void Form1_Load_1( object sender, EventArgs e) { //实例化委托 updateTxt = new UpdateTxt(UpdateTxtMethod); } |
UpdateTxtMethod方法如下
1 2 3 4 5 6 | //修改TextBox值的方法。 public void UpdateTxtMethod( string msg) { richTextBox1.AppendText(msg + "\r\n" ); richTextBox1.ScrollToCaret(); } |
下面我们来定义一个循环来输出一个值的,关调用这个委托来更新richTextBox1
1 2 3 4 5 6 7 8 9 10 11 12 | //此为在非创建线程中的调用方法,其实是使用TextBox的Invoke方法。 public void ThreadMethodTxt( int n) { this .BeginInvoke(updateTxt, "线程开始执行,执行" + n + "次,每一秒执行一次" ); for ( int i = 0; i < n; i++) { this .BeginInvoke(updateTxt, i.ToString()); //一秒 执行一次 Thread.Sleep(1000); } this .BeginInvoke(updateTxt, "线程结束" ); } |
然后就是使用一个子线程来调用它了
1 2 3 4 5 6 7 8 9 | //开启线程 private void button1_Click( object sender, EventArgs e) { Thread objThread = new Thread( new ThreadStart( delegate { ThreadMethodTxt(Convert.ToInt32(textBox1.Text.Trim())); })); objThread.Start(); } |
好了就这样基本上就可以了。
那问题现在那里呢,其实就出在这一句上
1 | this .BeginInvoke(updateTxt, "线程结束" ); |
大家也许已经发现了,我是这样写的,而不是
1 | updateTxt( "线程结束" ); |
这样来直接在子线程中使用,
我相信有很多同志都是这样写的,其实错就错在这里
如果直接使用
1 | updateTxt( "线程结束" ); |
大家想一下应该就明白了,
updateTxt是在主线程创建的,而我们在子线程中直接使用,运行的数据多了,就会出现卡死,这是界面信息堵死的原因,
所以就算是委托也不能直接在子线程中使用,而是要使用BeginInvoke方法来调用这个委托
这样才不会出现卡死的现像。
问题就解决了。