浅谈C#多线程与UI响应
www.educity.cn 发布者:shenywww 来源:网络转载 发布日期:2014年10月06日 文章评论 发表文章
一.C#多线程概述
1.后台循环任务,少量UI更新:例如批量上传文件,并提供进度。这种情况使用BackgroundWorker组件是非常好的选择。
2.耗时的后台任务:这里的耗时任务是指一个时间较长的任务,并且不能精确获取进度,如:调用一个远程WebService接口。这种情况可以开两个线程,一个工作,一个更新UI(不能提供进度,只能显示动画表示系统在运行中)。www.educity.cn 发布者:shenywww 来源:网络转载 发布日期:2014年10月06日 文章评论 发表文章
一.C#多线程概述
1.后台循环任务,少量UI更新:例如批量上传文件,并提供进度。这种情况使用BackgroundWorker组件是非常好的选择。
2.耗时的后台任务:这里的耗时任务是指一个时间较长的任务,并且不能精确获取进度,如:调用一个远程WebService接口。这种情况可以开两个线程,一个工作,一个更新UI(不能提供进度,只能显示动画表示系统在运行中)。
3.耗时的UI任务:当工作压力集中在UI响应上时,可以在工作者线程中增加延时,从而让UI线程获得响应时间。整个工作的总体时间会增加,但用户响应效果会好很多。
二.后台的循环任务,少量UI更新
这种情况使用BackgroundWorker组件是最好的选择。(详见附一)
三.后台耗时任务
在后台执行一个不可分解的耗时任务,需要进行界面更新,以便让客户看上去程序有所响应。这种情况下,UI线程一般也不知道工作线程何时结束,所以一般执行循环任务,当工作线程结束后,关闭UI线程就可以了。
Threaduithread=null; privatevoidbtnStart_Click(objectsender,EventArgse) { uithread=newThread(newThreadStart(this.UpdateProgressThread)); uithread.Start(); Threadworkthread=newThread(newThreadStart(this.DoSomething)); workthread.Start(); } privatevoidDoSomething() { Thread.Sleep(5000); uithread.Abort(); MessageBox.Show("workend"); } privatevoidUpdateProgressThread() { for(inti=0;i<10000;i++) { Thread.Sleep(100); this.Invoke(newAction(this.UpdateProgress),i); } } privatevoidUpdateProgress(intv) { this.progressBar1.Value=v; }
这里只要注意一点:线程调用的方法都不能访问用户控件,必须通过委托调用Form的方法来实现界面更新。
四.耗时的UI任务
当整个工作压力集中在UI响应上时,可以在工作者线程中增加延时,从而让UI线程获得响应时间。整个工作的总体时间会增加,但用户响应效果会好很多。
privatevoidFormInitForm_Load(objectsender,EventArgse) { this.listView1.Items.Clear(); Threadworkthread=newThread(newThreadStart(this.DoSomething)); workthread.Start(); } privatevoidDoSomething() { for(inti=0;i<30;i++) { this.Invoke(newAction(this.LoadPicture),i); Thread.Sleep(100); } } privatevoidLoadPicture(inti) { stringstringtext=string.Format("Item{0}",i); ListViewItemlvi=newListViewItem(text,0); this.listView1.Items.Add(lvi); Thread.Sleep(200);//模拟耗时UI任务,非循环,不可分解 }
五.补充
1.Invoke和BeginInvoke
在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过Invoke或者BeginInvoke去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。
而所谓的“一面响应操作,一面添加节点”永远只能是相对的,使UI线程的负担不至于太大而以,因为界面的正确更新始终要通过UI线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到UI线程中去做,这样也就达到了减轻UI线程负担的目的了。
2.Application.DoEvent
在耗时的循环的UI更新的方法中,插入Application.DoEvent,会使界面获得响应,Application.DoEvent会调用消息处理程序。
privatevoidbutton2_Click(objectsender,EventArgse) { for(inti=0;i<30;i++) { stringstringtext=string.Format("Item{0}",i); ListViewItemlvi=newListViewItem(text,0); this.listView1.Items.Add(lvi); Thread.Sleep(200); for(intj=0;j<10;j++) { Thread.Sleep(10); Application.DoEvents(); } } } 3.Lock lock(object) { } 等价与 try { Monitor.Enter(object); } finally { Monitor.Exit(object) }
附一:
BackgroundWorker组件使用说明
一.概述
BackgroundWorker是·NET2.0提供的一个多线程组件,在应用程序中使用,可以非常简单方便地实现UI控件通信,并自动处理多线程冲突问题。
二.基本属性
1.WorkerReportsProgress,bool:是否允许报告进度;
2.WorkerSupportsCancellation,bool:是否允许取消线程。
3.CancellationPending,bool,get:读取用户是否取消该线程。
三.基本事件
1.DoWork:工作者线程
2.RunWorkerCompleted:线程进度报告
3.ProgressChanged:线程结束报告
四.基本方法
1.RunWorkerAsync():启动工作者线程;
2.CancelAsync():取消工作者线程;
3.ReportProgress(int);报告进度
五.代码
//启动 privatevoidbtnStart_Click(objectsender,EventArgse) { this.btnStart.Enabled=false; this.btnStop.Enabled=true; this.backgroundWorker.RunWorkerAsync(); } //通知线程停止 privatevoidbtnStop_Click(objectsender,EventArgse) { this.backgroundWorker.CancelAsync(); } //工作者线程 privatevoidbackgroundWorker_DoWork(objectsender,DoWorkEventArgse) { for(inti=0;i<150;i++) { if(backgroundWorker.CancellationPending)//查看用户是否取消该线程 { break; } System.Threading.Thread.Sleep(50);//干点实际的事 backgroundWorker.ReportProgress(i);//报告进度 } } //线程进度报告 privatevoidbackgroundWorker_ProgressChanged(objectsender,ProgressChangedEventArgse) { this.progressBar1.Value=e.ProgressPercentage*100/150; } //线程结束报告 privatevoidbackgroundWorker_RunWorkerCompleted(objectsender,RunWorkerCompletedEventArgse) { this.btnStart.Enabled=true; this.btnStop.Enabled=false; }
3.耗时的UI任务:当工作压力集中在UI响应上时,可以在工作者线程中增加延时,从而让UI线程获得响应时间。整个工作的总体时间会增加,但用户响应效果会好很多。
二.后台的循环任务,少量UI更新
这种情况使用BackgroundWorker组件是最好的选择。(详见附一)
三.后台耗时任务
在后台执行一个不可分解的耗时任务,需要进行界面更新,以便让客户看上去程序有所响应。这种情况下,UI线程一般也不知道工作线程何时结束,所以一般执行循环任务,当工作线程结束后,关闭UI线程就可以了。
Threaduithread=null; privatevoidbtnStart_Click(objectsender,EventArgse) { uithread=newThread(newThreadStart(this.UpdateProgressThread)); uithread.Start(); Threadworkthread=newThread(newThreadStart(this.DoSomething)); workthread.Start(); } privatevoidDoSomething() { Thread.Sleep(5000); uithread.Abort(); MessageBox.Show("workend"); } privatevoidUpdateProgressThread() { for(inti=0;i<10000;i++) { Thread.Sleep(100); this.Invoke(newAction(this.UpdateProgress),i); } } privatevoidUpdateProgress(intv) { this.progressBar1.Value=v; }
这里只要注意一点:线程调用的方法都不能访问用户控件,必须通过委托调用Form的方法来实现界面更新。
四.耗时的UI任务
当整个工作压力集中在UI响应上时,可以在工作者线程中增加延时,从而让UI线程获得响应时间。整个工作的总体时间会增加,但用户响应效果会好很多。
privatevoidFormInitForm_Load(objectsender,EventArgse) { this.listView1.Items.Clear(); Threadworkthread=newThread(newThreadStart(this.DoSomething)); workthread.Start(); } privatevoidDoSomething() { for(inti=0;i<30;i++) { this.Invoke(newAction(this.LoadPicture),i); Thread.Sleep(100); } } privatevoidLoadPicture(inti) { stringstringtext=string.Format("Item{0}",i); ListViewItemlvi=newListViewItem(text,0); this.listView1.Items.Add(lvi); Thread.Sleep(200);//模拟耗时UI任务,非循环,不可分解 }
五.补充
1.Invoke和BeginInvoke
在多线程编程中,我们经常要在工作线程中去更新界面显示,而在多线程中直接调用界面控件的方法是错误的做法,正确的做法是将工作线程中涉及更新界面的代码封装为一个方法,通过Invoke或者BeginInvoke去调用,两者的区别就是一个导致工作线程等待,而另外一个则不会。
而所谓的“一面响应操作,一面添加节点”永远只能是相对的,使UI线程的负担不至于太大而以,因为界面的正确更新始终要通过UI线程去做,我们要做的事情是在工作线程中包揽大部分的运算,而将对纯粹的界面更新放到UI线程中去做,这样也就达到了减轻UI线程负担的目的了。
2.Application.DoEvent
在耗时的循环的UI更新的方法中,插入Application.DoEvent,会使界面获得响应,Application.DoEvent会调用消息处理程序。
privatevoidbutton2_Click(objectsender,EventArgse) { for(inti=0;i<30;i++) { stringstringtext=string.Format("Item{0}",i); ListViewItemlvi=newListViewItem(text,0); this.listView1.Items.Add(lvi); Thread.Sleep(200); for(intj=0;j<10;j++) { Thread.Sleep(10); Application.DoEvents(); } } } 3.Lock lock(object) { } 等价与 try { Monitor.Enter(object); } finally { Monitor.Exit(object) }
附一:
BackgroundWorker组件使用说明
一.概述
BackgroundWorker是·NET2.0提供的一个多线程组件,在应用程序中使用,可以非常简单方便地实现UI控件通信,并自动处理多线程冲突问题。
二.基本属性
1.WorkerReportsProgress,bool:是否允许报告进度;
2.WorkerSupportsCancellation,bool:是否允许取消线程。
3.CancellationPending,bool,get:读取用户是否取消该线程。
三.基本事件
1.DoWork:工作者线程
2.RunWorkerCompleted:线程进度报告
3.ProgressChanged:线程结束报告
四.基本方法
1.RunWorkerAsync():启动工作者线程;
2.CancelAsync():取消工作者线程;
3.ReportProgress(int);报告进度
五.代码
//启动 privatevoidbtnStart_Click(objectsender,EventArgse) { this.btnStart.Enabled=false; this.btnStop.Enabled=true; this.backgroundWorker.RunWorkerAsync(); } //通知线程停止 privatevoidbtnStop_Click(objectsender,EventArgse) { this.backgroundWorker.CancelAsync(); } //工作者线程 privatevoidbackgroundWorker_DoWork(objectsender,DoWorkEventArgse) { for(inti=0;i<150;i++) { if(backgroundWorker.CancellationPending)//查看用户是否取消该线程 { break; } System.Threading.Thread.Sleep(50);//干点实际的事 backgroundWorker.ReportProgress(i);//报告进度 } } //线程进度报告 privatevoidbackgroundWorker_ProgressChanged(objectsender,ProgressChangedEventArgse) { this.progressBar1.Value=e.ProgressPercentage*100/150; } //线程结束报告 privatevoidbackgroundWorker_RunWorkerCompleted(objectsender,RunWorkerCompletedEventArgse) { this.btnStart.Enabled=true; this.btnStop.Enabled=false; }
浅谈C#多线程与UI响应的更多相关文章
- 浅谈iOS多线程
浅谈iOS多线程 首先,先看看进程和线程的概念. 图1.1 这一块不难理解,重点点下他们的几个重要区别: 1,地址空间和资源:进程可以申请和拥有系统资源,线程不行.资源进程间相互独立,同一进程的各线程 ...
- C#多线程与UI响应 跨线程更新UI
最近在写一个TCP通信程序,自定义了一个通信类TCPclient,用于客户端异步接收和发送网络消息. TCPclient中定义了一个接收到新的网络消息事件: //收到新消息事件 public dele ...
- 转载:浅谈Java多线程的同步问题【很好我就留下来,多分共享】
转载:http://www.cnblogs.com/phinecos/archive/2010/03/13/1684877.html#undefined 多线程的同步依靠的是对象锁机制,synchro ...
- 浅谈Java多线程的同步问题 【转】
多线程的同步依靠的是对象锁机制,synchronized关键字的背后就是利用了封锁来实现对共享资源的互斥访问. 下面以一个简单的实例来进行对比分析.实例要完成的工作非常简单,就是创建10个线程,每个线 ...
- 浅谈Java多线程同步机制之同步块(方法)——synchronized
在多线程访问的时候,同一时刻只能有一个线程能够用 synchronized 修饰的方法或者代码块,解决了资源共享.下面代码示意三个窗口购5张火车票: package com.jikexueyuan.t ...
- 浅谈Java多线程
线程与进程 什么是进程? 当一个程序进入内存中运行起来它就变为一个进程.因此,进程就是一个处于运行状态的程序.同时进程具有独立功能,进程是操作系统进行资源分配和调度的独立单位. 什么是线程? 线程是进 ...
- 浅谈 Java 多线程(一) --- JMM
为什么使用多线程 更多的处理器核心数(硬件的发展使 CPU 趋向于更多的核心数,如果不能充分利用,就无法显著提升程序的效率) 更快的响应时间(复杂的业务场景下,会存在许多数据一致性不强的操作,如果将这 ...
- 浅谈Java多线程中的join方法
先上代码 新建一个Thread,代码如下: package com.thread.test; public class MyThread extends Thread { private String ...
- 【转】浅谈多核CPU、多线程、多进程
浅谈多核CPU.多线程.多进程 1.CPU发展趋势 核心数目依旧会越来越多,依据摩尔定律,由于单个核心性能提升有着严重的瓶颈问题,普通的桌面PC有望在2017年末2018年初达到24核心(或者16核3 ...
随机推荐
- 解决Mysql错误Too many connections的方法
MySQL数据库 Too many connections出现这种错误明显就是 mysql_connect 之后忘记 mysql_close:当大量的connect之后,就会出现Too many co ...
- 玩转VFS(二)
关于VFS的第一篇中已经太长了 http://www.cnblogs.com/honpey/p/6348914.html 另起一篇: 1)如何在kernel里找到目前文件系统中的根目录: 2) 如何能 ...
- JavaScript内置对象常用
Math 提供了数学中常用的属性和方法,使用时直接用Math.属性/方法,而不需要new一个Math对象 Date 使用Date对象来对日期和时间进行操作.使用时,必须用new创建一个实例 windo ...
- Java进阶
Java进阶(一)Annotation(注解) Java进阶(二)当我们说线程安全时,到底在说什么 Java进阶(三)多线程开发关键技术 Java进阶(四)线程间通信方式对比 Java进阶(五)NIO ...
- TCP/IP Note3
TCP/IP协议 TCP/IP是不同的通信协议的大集合. 协议族:TCP/IP是基于TCP和IP这两个最初的协议之上的不同的通信协议的大集合. 1. TCP - 传输控制协议 TCP用于从应用程序到网 ...
- 2017 Multi-University Training Contest - Team 4 phone call(树+lca+并查集)
题解: (并查集处理往上跳的时候,一定要先让u,v往上跳到并查集的祖先,不然会wa掉) 代码如下: #include <iostream> #include <algorithm&g ...
- 左侧导航条+中间显示内容+右侧菜单配置,Bootstrap+AdminLTE+Jquery
1.最近做个导航页面,找了一大堆UI,最终选了AdminLTE,这个UI也是以bootstrap为基础,简单实用,中间内容用jquery的load加载,简单暴力,非常适合快速开发. 2.效果图如下: ...
- BZOJ1407 [Noi2002]Savage 【扩展欧几里得】
题目链接 BZOJ1407 题解 枚举\(m\)用扩欧判即可 #include<algorithm> #include<iostream> #include<cstrin ...
- swagger学习2
转:http://blog.csdn.net/fansunion/article/details/51923720 写的非常好,非常详细,推荐!!!! 最常用的5个注解 @Api:修饰整个类,描述Co ...
- git上传本地项目
1.(先进入项目文件夹)通过命令 git init 把这个目录变成git可以管理的仓库 git init 2.把文件添加到版本库中,使用命令 git add .添加到暂存区里面去,不要忘记后面的小数点 ...