LOGO OA教程 ERP教程 模切知识交流 PMS教程 CRM教程 开发文档 其他文档  
 
网站管理员

[点晴永久免费OA]C#多线程解决界面卡死问题的完美解决方案

admin
2019年10月25日 18:10 本文热度 3517

问题描述:
当我们的界面需要在程序运行中不断更新数据时,当一个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方法来调用这个委托
这样才不会出现卡死的现像。
问题就解决了。


该文章在 2019/10/25 18:18:39 编辑过
关键字查询
相关文章
正在查询...
点晴ERP是一款针对中小制造业的专业生产管理软件系统,系统成熟度和易用性得到了国内大量中小企业的青睐。
点晴PMS码头管理系统主要针对港口码头集装箱与散货日常运作、调度、堆场、车队、财务费用、相关报表等业务管理,结合码头的业务特点,围绕调度、堆场作业而开发的。集技术的先进性、管理的有效性于一体,是物流码头及其他港口类企业的高效ERP管理信息系统。
点晴WMS仓储管理系统提供了货物产品管理,销售管理,采购管理,仓储管理,仓库管理,保质期管理,货位管理,库位管理,生产管理,WMS管理系统,标签打印,条形码,二维码管理,批号管理软件。
点晴免费OA是一款软件和通用服务都免费,不限功能、不限时间、不限用户的免费OA协同办公管理系统。
Copyright 2010-2025 ClickSun All Rights Reserved