线程生命周期(来源 w3cschool)

  1. 未启动状态:当线程实例被创建但 Start 方法未被调用时的状况。
  2. 就绪状态:当线程准备好运行并等待 CPU 周期时的状况。
  3. 不可运行状态
    • 已经调用 Sleep 方法
    • 已经调用 Wait 方法
    • 通过 I/O 操作阻塞
  4. 死亡状态:当线程已完成执行或已中止时的状况。

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#] 多线程总结(结合进度条)的更多相关文章

  1. (委托事件处理)关于多线程执行显示进度条的实例(转)&&线程间操作无效: 从不是创建控件“rtxtEntryNO”的线程访问它。

    关于多线程执行显示进度条的实例! 之前回答了一篇关于怎么在线程中操作进度条的帖子,估计有人看的不是很明白今天没事,写了一个小小的实例,很简单,就2个文件权当抛砖引玉,希望有更好解决方案的人发表一下意见 ...

  2. C#编程总结(四)多线程应用(进度条的编程问题)——转自http://www.cnblogs.com/yank/p/3232955.html

    多线程应用 多线程应用很广泛,简单总结了一下: 1)不阻断主线程,实现即时响应,由后台线程完成特定操作2)多个线程,完成同类任务,提高并发性能3)一个任务有多个独立的步骤,多个线程并发执行各子任务,提 ...

  3. 赵雅智_android多线程下载带进度条

    progressBar说明 在某些操作的进度中的可视指示器,为用户呈现操作的进度,还它有一个次要的进度条,用来显示中间进度,如在流媒体播放的缓冲区的进度. 一个进度条也可不确定其进度.在不确定模式下, ...

  4. Python:如何显示进度条

    首先,推荐一个组件:progressive 效果如下: 进度条和一般的print区别在哪里呢? 答案就是print会输出一个\n,也就是换行符,这样光标移动到了下一行行首,接着输出,之前已经通过std ...

  5. C# 通过线程来控制进度条(转)--讲解多线程对界面的操作

    // 通过创建委托解决传递参数问题 private void _btnRun_Click( object sender, System.EventArgs e ) { RunTaskDelegate ...

  6. C# 通过委托控制进度条以及多线程更新控件

    using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; usin ...

  7. Android开发之多线程下载、断点续传、进度条和文本显示

    代码实现了在Android环境下的多线程下载.断点续传.进度条显示和文本显示百分数: import java.io.BufferedReader; import java.io.File; impor ...

  8. ASP.NET 多线程 监控任务执行情况,并显示进度条

    关于多线程的基本概念和知识在本文中不多讲,而且我懂的也不是很透,说的太多误人子弟...对于我来说,做本文提到的功能够用就行,等实现其他效果不够用的时候,再深入研究 推荐看园子里的两篇博客应该就有个基本 ...

  9. WPF多线程下载文件,有进度条

    //打开对话框选择文件         private void OpenDialogBox_Click(object sender, RoutedEventArgs e)         {     ...

随机推荐

  1. 谈谈一些有趣的CSS题目(四)-- 从倒影说起,谈谈 CSS 继承 inherit

    开本系列,讨论一些有趣的 CSS 题目,抛开实用性而言,一些题目为了拓宽一下解决问题的思路,此外,涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题中有你感觉 ...

  2. P2V之后的磁盘扩容新思路

    背景: 原先的物理机环境多是若干块物理磁盘经过RAID卡进行了RAID5之后的虚拟磁盘组,这样我们在操作系统内看到的也就是一块完整的磁盘.我们会在上面进行分区,然后格式化后以便使用. Figure 1 ...

  3. Mysql 忘记root密码处理办法

    一.更改my.cnf配置文件 1.用命令编辑/etc/my.cnf配置文件,即:vim /etc/my.cnf 或者 vi /etc/my.cnf 2.在[mysqld]下添加skip-grant-t ...

  4. 初尝Brnshop移植到Linux Mono Jexus环境运行

    brnshop是最近社区上比较火的开源商城. Jexus是Linux上的web服务器,简单说就是Linux的iis吧.特别感谢作者宇内流云的指点 一.根据http://www.cnblogs.com/ ...

  5. lucene 基础知识点

    部分知识点的梳理,参考<lucene实战>及网络资料 1.基本概念 lucence 可以认为分为两大组件: 1)索引组件 a.内容获取:即将原始的内容材料,可以是数据库.网站(爬虫).文本 ...

  6. 转 10 个最佳的 Node.js 的 MVC 框架

    10 个最佳的 Node.js 的 MVC 框架 oschina 发布于: 2014年02月24日 (33评) 分享到:    收藏 +322 Node.js 是一个基于Chrome JavaScri ...

  7. 架构 Roadmap 笔记分享

    虽然我们的架构不是开源的,不过一些笔记可以愿意公开和大家讨论一下,我相信不少人在和我们干着同样的事情,那不如一块儿交流一下,这样我们可以更快. 这里前端,后端都有,前端我们用的是 avalon js, ...

  8. Entity Framework 6 Recipes 2nd Edition(11-4)译 -> 在”模型定义”函数里调用另一个”模型定义”函数

    11-4.在”模型定义”函数里调用另一个”模型定义”函数 问题 想要用一个”模型定义”函数去实现另一个”模型定义”函数 解决方案 假设我们已有一个公司合伙人关系连同它们的结构模型,如Figure 11 ...

  9. Android开发学习之路-3DTouch效果模仿

    3D Touch是什么效果的大家应该都知道了.什么?不知道,那也没办法呀,我也没有iPhone 6s演示给你看的. 本篇博客要做的效果图: 来个低质量动图: 这个动图效果不是很好,实际上模糊效果应该是 ...

  10. 有关“数据统计”的一些概念 -- PV UV VV IP跳出率等

    有关"数据统计"的一些概念 -- PV UV VV IP跳出率等 版权声明:本文为博主原创文章,未经博主允许不得转载. 此文是本人工作中碰到的,随时记下来的零散概念,特此整理一下. ...