zoukankan      html  css  js  c++  java
  • C#多线程解决界面卡死问题的完美解决方案,BeginInvoke而不是委托delegate

     问题描述:
    当我们的界面需要在程序运行中不断更新数据时,当一个textbox的数据需要变化时,为了让程序执行中不出现界面卡死的现像,最好的方法就是多线程来解决
    一个主线程来创建界面,使用一个子线程来执行程序并更新主界面
    这样就不会出现卡死的现像了
    这肯定是没有问题的,
    但是为什么在使用的过程中一样会有很多地方会出现卡死呢,而且有用户跟我说是我的Httphelper类的问题,其实不是,而且我再次声明我的Httphelper类跟多线程并没有关系。不要在诬赖我了哦。
    这个问题其实也困或了我很久,但是今天终于解决了,而且我发现很多人有这样的问题,所以我分享一个例子方便大家参考吧。
    先来看看我的界面

    当我单击
    开始执行后

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

     1 using System;
     2 using System.Collections.Generic;
     3 using System.ComponentModel;
     4 using System.Data;
     5 using System.Drawing;
     6 using System.Linq;
     7 using System.Text;
     8 using System.Windows.Forms;
     9 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 + "
    ");
    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 }

    就这些代码,大家看注释应该就明白一点了,
    主要是使用一个委托来更新界面的richTextBox1
    这样写是肯定没有问题的,而且在我其它的更高级一点的例子里也是这么写的
    C#多线程|匿名委托传参数|测试网站压力--升级版
    http://www.sufeinet.com/thread-13-1-1.html
    上面的文件大家可以做为参考
    那问题现在那里呢,其实就出在这一句上

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

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

    updateTxt("线程结束");

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

    updateTxt("线程结束");

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

    原文:http://www.sufeinet.com/thread-3556-1-1.html

  • 相关阅读:
    Pycharm如何自动换行
    Android逆向基础
    动态调试ELF文件Crackme
    用Hash 算法给payload瘦身
    010 editor手写pe文件
    win脱壳_压缩壳_aspack
    WannaCrypt0r分析报告
    JVM内存分区
    java设计模式(模板方法模式)
    es6
  • 原文地址:https://www.cnblogs.com/RCJL/p/12826668.html
Copyright © 2011-2022 走看看