C#+.NETFramwork开发1:多线程

C#+.NETFramwork开发多线程解决界面卡死问题的完美解决方案

参考网址:**C#多线程解决界面卡死问题的完美解决方案

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

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

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);
        }
    }
}

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

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

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

updateTxt("线程结束")

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

updateTxt("线程结束")

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

C#中使用多线程

C#调用线程无法访问此对象,因为另一个线程拥有该对象.–解决方法

参考网址C#调用线程无法访问此对象,因为另一个线程拥有该对象.–解决方法

错误复现

// 主线程定义了一个对象
Label lb = new Label();
// 子线程调用该对象即会报此错误
lb.Content = "111";

原因是C#中不允许多个线程同时拥有操控一个对象的权利

解决办法

利用委托机制, 子线程向主线程发起一个委托, 由主线程触发相关动作帮助我们完成操作

// 委托
lb.Dispatcher.Invoke(new Action(
	delegate 
	{ 
		// ------- 需要完成什么操作,写在这里就可以了, 主线程会触发该Action来完成-------
		lb.Content = "666"; 
	})
	);

C#多线程开发:线程挂起,恢复与中止等操作

Thread workThread = new Thread(new ThreadStart(CalcSum));//初始化线程
Thread workThread = new Thread(new ThreadStart(CalcSum));//启动线程
workThread.Suspend();//线程挂起(暂停)
workThread.Resume();//线程恢复(继续)
workThread.Abort();//线程结束
workThread.Sleep(10000);//线程暂停10000毫秒
workThread.join();//join表示需要等待当前线程结束才可进行主线程
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值