分享

C#多线程解决界面卡死问题的完美解决方案 – 放肆雷特 | 胖子的技术博客

 hjqh886 2013-10-18

文章下最方有源码下载
问题描述:
当我们的界面需要在程序运行中不断更新数据时,当一个textbox的数据需要变化时。
为了让程序执行中不出现界面卡死的现像,最好的方法就是多线程来解决,一个主线程来创建界面,使用一个子线程来执行程序并更新主界面,这样就不会出现卡死的现像了。
但是为什么在使用的过程中一样会有很多地方会出现卡死呢?这个问题其实也困或了我很久,但是今天终于解决了,而且我发现很多人有这样的问题,所以我分享一个例子方便大家参考。
先来看看我的界面

当我单击开始执行后

是数据在不断的更新
这个时候界面是不会卡死的,只是数据在不断的更新
下面看看我的代码

01 using System;
02 using System.Collections.Generic;
03 using System.ComponentModel;
04 using System.Data;
05 using System.Drawing;
06 using System.Linq;
07 using System.Text;
08 using System.Windows.Forms;
09 using System.Threading;
10  
11 namespace WindowsFormsApplication3
12 {
13     public partial class Form1 : Form
14     {
15         public Form1()
16         {
17             InitializeComponent();
18         }
19         //创建一个委托,是为访问TextBox控件服务的。
20         public delegate void UpdateTxt(string msg);
21         //定义一个委托变量
22         public UpdateTxt updateTxt;
23  
24         //修改TextBox值的方法。
25         public void UpdateTxtMethod(string msg)
26         {
27             richTextBox1.AppendText(msg + "\r\n");
28             richTextBox1.ScrollToCaret();
29         }
30  
31         //此为在非创建线程中的调用方法,其实是使用TextBox的Invoke方法。
32         public void ThreadMethodTxt(int n)
33         {
34             this.BeginInvoke(updateTxt, "线程开始执行,执行" + n + "次,每一秒执行一次");
35             for (int i = 0; i < n; i++)
36             {
37                 this.BeginInvoke(updateTxt, i.ToString());
38                 //一秒 执行一次
39                 Thread.Sleep(1000);
40             }
41             this.BeginInvoke(updateTxt, "线程结束");
42         }
43         //开启线程
44         private void button1_Click(object sender, EventArgs e)
45         {
46             Thread objThread = new Thread(new ThreadStart(delegate
47             {
48                 ThreadMethodTxt(Convert.ToInt32(textBox1.Text.Trim()));
49             }));
50             objThread.Start();
51         }
52  
53         private void Form1_Load_1(object sender, EventArgs e)
54         {
55             //实例化委托
56             updateTxt = new UpdateTxt(UpdateTxtMethod);
57         }
58     }
59 }

上面是全部代码方便大家参考吧

第一步我们先来定义一个委托updateTxt

1 //创建一个委托,是为访问TextBox控件服务的。
2 public delegate void UpdateTxt(string msg);
3 //定义一个委托变量
4 public UpdateTxt updateTxt;

主要是使用一个委托来更新界面的richTextBox1
实例方法如下

1 private void Form1_Load_1(object sender, EventArgs e)
2 {
3     //实例化委托
4     updateTxt = new UpdateTxt(UpdateTxtMethod);
5 }

UpdateTxtMethod方法如下

1 //修改TextBox值的方法。
2 public void UpdateTxtMethod(string msg)
3 {
4     richTextBox1.AppendText(msg + "\r\n");
5     richTextBox1.ScrollToCaret();
6 }

下面我们来定义一个循环来输出一个值的,关调用这个委托来更新richTextBox1

01 //此为在非创建线程中的调用方法,其实是使用TextBox的Invoke方法。
02 public void ThreadMethodTxt(int n)
03 {
04     this.BeginInvoke(updateTxt, "线程开始执行,执行" + n + "次,每一秒执行一次");
05     for (int i = 0; i < n; i++)
06     {
07         this.BeginInvoke(updateTxt, i.ToString());
08         //一秒 执行一次
09         Thread.Sleep(1000);
10     }
11     this.BeginInvoke(updateTxt, "线程结束");
12 }

然后就是使用一个子线程来调用它了

1 //开启线程
2 private void button1_Click(object sender, EventArgs e)
3 {
4     Thread objThread = new Thread(new ThreadStart(delegate
5     {
6         ThreadMethodTxt(Convert.ToInt32(textBox1.Text.Trim()));
7     }));
8     objThread.Start();
9 }

好了就这样基本上就可以了。

那问题现在那里呢,其实就出在这一句上

1 this.BeginInvoke(updateTxt, "线程结束");

大家也许已经发现了,我是这样写的,而不是

1 updateTxt("线程结束");

这样来直接在子线程中使用,
我相信有很多同志都是这样写的,其实错就错在这里
如果直接使用

1 updateTxt("线程结束");

大家想一下应该就明白了,
updateTxt是在主线程创建的,而我们在子线程中直接使用,运行的数据多了,就会出现卡死,这是界面信息堵死的原因,
所以就算是委托也不能直接在子线程中使用,而是要使用BeginInvoke方法来调用这个委托
这样才不会出现卡死的现像。
问题就解决了。
大家支持一下哦
下面是我的源码提供给大家下载吧 猛击下载

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多