[C#] 多线程总结(结合进度条)
线程生命周期(来源 w3cschool)
- 未启动状态:当线程实例被创建但 Start 方法未被调用时的状况。
- 就绪状态:当线程准备好运行并等待 CPU 周期时的状况。
- 不可运行状态:
- 已经调用 Sleep 方法
- 已经调用 Wait 方法
- 通过 I/O 操作阻塞
- 死亡状态:当线程已完成执行或已中止时的状况。
Thread 常用方法:
- public void Interrupt() 中断处于 WaitSleepJoin 线程状态的线程。
- public void Join() 在继续执行标准的 COM 和 SendMessage 消息泵处理期间,阻塞调用线程,直到某个线程终止为止。
- public void Start() 开始一个线程
- public static void Sleep(int millisecondsTimeout) 让线程暂停一段时间
一 普通线程
分为两种,一种是不需要给子线程传参数,Thread t = new Thread(new ThreadStart(void () target)); 另一种是要给子线程传一个参数,Thread t = new Thread(new ParameterizedThreadStart(void (object) target));
// 普通线程
private void btn1_Click(object sender, EventArgs e)
{
progressBar.Value = 0;
Thread tt = new Thread(new ThreadStart(DoWork1));
tt.Name = "不带参数普通线程";
tt.Start();
Thread t = new Thread(new ParameterizedThreadStart(DoWork2));
t.Name = "带参数普通线程";
t.IsBackground = true;
t.Start(100);
_msg += "当前线程的执行状态:" + t.IsAlive + "\r\n";
_msg += "当前托管线程的唯一标识:" + t.ManagedThreadId + "\r\n";
_msg += "线程名称:" + t.Name + "\r\n";
_msg += "当前线程的状态:" + t.ThreadState;
MessageBox.Show("消息:\r\n" + _msg, "提示", MessageBoxButtons.OK);
}
// 线程方法
private void DoWork1()
{
for (int i = 0; i < 100; i++)
{
// 跨线程访问 UI,BeginInvoke 采用异步委托
progressBar.BeginInvoke(new EventHandler((sender, e) =>
{
progressBar.Value = i;
}), null);
}
}
// 线程方法
private void DoWork2(object obj)
{
for (int i = 0; i < (int)obj; i++)
{
progressBar.BeginInvoke(new EventHandler((sender, e) =>
{
progressBar.Value = i;
}), null);
}
}
普通线程
二 线程池
public static bool QueueUserWorkItem(WaitCallback);
public static bool QueueUserWorkItem(WaitCallback, object);
线程池默认为后台线程(IsBackground)
private void btn3_Click(object sender, EventArgs e)
{
ThreadPool.QueueUserWorkItem(DoWork2, 100);
// 或者
ThreadPool.QueueUserWorkItem((s) =>
{
int minWorkerThreads, minCompletionPortThreads, maxWorkerThreads, maxCompletionPortThreads;
ThreadPool.GetMinThreads(out minWorkerThreads, out minCompletionPortThreads);
ThreadPool.GetMaxThreads(out maxWorkerThreads, out maxCompletionPortThreads);
MessageBox.Show(String.Format("WorkerThreads = {0} ~ {1}, CompletionPortThreads = {2} ~ {3}",
minWorkerThreads, maxWorkerThreads, minCompletionPortThreads, maxCompletionPortThreads));
DoWork2(100);
});
}
// 线程方法
private void DoWork2(object obj)
{
for (int i = 0; i < (int)obj; i++)
{
// Thread.Sleep(50);
progressBar.BeginInvoke(new EventHandler((sender, e) =>
{
progressBar.Value = i;
}), null);
}
}
线程池
三 BackgroundWorker
private void btn4_Click(object sender, EventArgs e)
{
progressBar.Value = 0;
BackgroundWorker bw = new BackgroundWorker();
bw.WorkerReportsProgress = true;// 是否报告进度更新
// 线程执行
bw.DoWork += new DoWorkEventHandler((obj, args) =>
{
for (int i = 0; i < 100; i++)
{
bw.ReportProgress(i);
}
});
// UI主线程显示进度
bw.ProgressChanged += (obj, progressChangedEventArgs) =>
{
progressBar.Value = progressChangedEventArgs.ProgressPercentage;
};
// 线程执行完成后的回调函数
bw.RunWorkerCompleted += (obj, runWorkerCompletedEventArgs) =>
{
MessageBox.Show("子线程执行完成!");
};
if (!bw.IsBusy)
{
bw.RunWorkerAsync();
}
}
BackgroundWorker
三 Task(.NET 4.0以上版本)
参考博客 http://www.cnblogs.com/luxiaoxun/p/3280146.html
private void btn5_Click(object sender, EventArgs e)
{
progressBar.Value = 0;
Task<bool> t = new Task<bool>(maxValue => DoWork((int)maxValue), progressBar.Maximum);
t.Start();
t.Wait();
// 任务完成后继续延续任务
Task cwt = t.ContinueWith(task => MessageBox.Show("The result is " + t.Result));
}
// 线程方法
private bool DoWork(int maxValue)
{
for (int n = 0; n < maxValue; n++)
{
progressBar.BeginInvoke(new EventHandler((sender, e) =>
{
progressBar.Value = n;
}), null);
} return true;
}
Task
四 异步委托
参考博客 http://www.cnblogs.com/luxiaoxun/p/3280146.html
public delegate string MyDelegate(object arg); private void btn6_Click(object sender, EventArgs e)
{
MyDelegate myDelegate = new MyDelegate(DoWork3);
IAsyncResult result = myDelegate.BeginInvoke(100, DoWork2Callback, "回调函数参数"); // 异步执行完成
string resultStr = myDelegate.EndInvoke(result);
} // 线程函数
private string DoWork3(object arg)
{
for (int n = 0; n < (int)arg; n++)
{
progressBar.BeginInvoke(new EventHandler((sender, e) =>
{
progressBar.Value = n;
}), null);
} return "Finished";
} // 异步回调函数
private void DoWork2Callback(IAsyncResult arg)
{
MessageBox.Show(arg.AsyncState.ToString());
}
异步委托
五 附 跨线程访问UI之 SynchronizationContext (同步上下文)
private void btn2_Click(object sender, EventArgs e)
{
SynchronizationContext context = SynchronizationContext.Current;
new Thread(() =>
{
for (int i = 0; i < 100; i++)
{
// Send方法是发送一个异步请求消息
//context.Send((s) =>
//{
// progressBar.Value = i;
//}, null);
// Post方法是发送一个同步请求消息
context.Post((s) =>
{
progressBar.Value = i;
}, null);
}
}).Start();
}
SynchronizationContext
六 参考资料:
☆多线程讲解 http://www.w3cschool.cc/csharp/csharp-multithreading.html
☆http://www.cnblogs.com/luxiaoxun/p/3280146.html
[C#] 多线程总结(结合进度条)的更多相关文章
- (委托事件处理)关于多线程执行显示进度条的实例(转)&&线程间操作无效: 从不是创建控件“rtxtEntryNO”的线程访问它。
关于多线程执行显示进度条的实例! 之前回答了一篇关于怎么在线程中操作进度条的帖子,估计有人看的不是很明白今天没事,写了一个小小的实例,很简单,就2个文件权当抛砖引玉,希望有更好解决方案的人发表一下意见 ...
- C#编程总结(四)多线程应用(进度条的编程问题)——转自http://www.cnblogs.com/yank/p/3232955.html
多线程应用 多线程应用很广泛,简单总结了一下: 1)不阻断主线程,实现即时响应,由后台线程完成特定操作2)多个线程,完成同类任务,提高并发性能3)一个任务有多个独立的步骤,多个线程并发执行各子任务,提 ...
- 赵雅智_android多线程下载带进度条
progressBar说明 在某些操作的进度中的可视指示器,为用户呈现操作的进度,还它有一个次要的进度条,用来显示中间进度,如在流媒体播放的缓冲区的进度. 一个进度条也可不确定其进度.在不确定模式下, ...
- Python:如何显示进度条
首先,推荐一个组件:progressive 效果如下: 进度条和一般的print区别在哪里呢? 答案就是print会输出一个\n,也就是换行符,这样光标移动到了下一行行首,接着输出,之前已经通过std ...
- C# 通过线程来控制进度条(转)--讲解多线程对界面的操作
// 通过创建委托解决传递参数问题 private void _btnRun_Click( object sender, System.EventArgs e ) { RunTaskDelegate ...
- C# 通过委托控制进度条以及多线程更新控件
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...
- Android开发之多线程下载、断点续传、进度条和文本显示
代码实现了在Android环境下的多线程下载.断点续传.进度条显示和文本显示百分数: import java.io.BufferedReader; import java.io.File; impor ...
- ASP.NET 多线程 监控任务执行情况,并显示进度条
关于多线程的基本概念和知识在本文中不多讲,而且我懂的也不是很透,说的太多误人子弟...对于我来说,做本文提到的功能够用就行,等实现其他效果不够用的时候,再深入研究 推荐看园子里的两篇博客应该就有个基本 ...
- WPF多线程下载文件,有进度条
//打开对话框选择文件 private void OpenDialogBox_Click(object sender, RoutedEventArgs e) { ...
随机推荐
- [.NET] C# 知识回顾 - Event 事件
C# 知识回顾 - Event 事件 [博主]反骨仔 [原文]http://www.cnblogs.com/liqingwen/p/6060297.html 序 昨天,通过<C# 知识回顾 - ...
- Git小技巧 - 指令别名及使用Beyond Compare作为差异比较工具
前言 本文主要写给使用命令行来操作Git的用户,用于提高Git使用的效率.至于使用命令还是GUI(Tortoise Git或VS的Git插件)就不在此讨论了,大家根据自己的的喜好选择就好.我个人是比较 ...
- 关于MJRefresh的下拉加载数据bug
当没有更多数据的时候显示NoMoreData 我的理解是先结束刷新再显示没有更多 今天之前一直没发现有问题 贴之前的代码 [self.collectionView reloadData]; [self ...
- git init和git init -bare区别
1 Git init 和 git init –bare 的区别 用"git init"初始化的版本库用户也可以在该目录下执行所有git方面的操作.但别的用户在将更新push上来的 ...
- 【转】组件化的Web王国
本文由 埃姆杰 翻译.未经许可,禁止转载!英文出处:Future Insights. 内容提要 使用许多独立组件构建应用程序的想法并不新鲜.Web Component的出现,是重新回顾基于组件的应用程 ...
- [Hadoop in Action] 第7章 细则手册
向任务传递定制参数 获取任务待定的信息 生成多个输出 与关系数据库交互 让输出做全局排序 1.向任务传递作业定制的参数 在编写Mapper和Reducer时,通常会想让一些地方可以配 ...
- Linux目录结构
- BZOJ 2119: 股市的预测 [后缀数组 ST表]
2119: 股市的预测 Time Limit: 10 Sec Memory Limit: 259 MBSubmit: 331 Solved: 153[Submit][Status][Discuss ...
- mono for android学习过程系列教程(5)
这一讲主要需要了解的安卓UI元素是Spinner.这个元素类似我们 winform和webform里面的下拉选项. 首先我们先建立一个新的项目,命名为SpinnerExample. 然后在Layout ...
- Netty构建分布式消息队列实现原理浅析
在本人的上一篇博客文章:Netty构建分布式消息队列(AvatarMQ)设计指南之架构篇 中,重点向大家介绍了AvatarMQ主要构成模块以及目前存在的优缺点.最后以一个生产者.消费者传递消息的例子, ...