C#多线程实现方法——线程池(Thread Pool)
ThreadPool使用
需要定义waitcallback委托形式如
- public delegate void WaitCallback(object state);
public delegate void WaitCallback(object state);
例如如下例子:
- static private void ThreadWorker(object state)
- {
- int num = (int)state;
- if (num > 0)
- {
- for (int i = 0; i < 3; i++)
- {
- Thread.Sleep(1000);// sleep for one second
- num++;
- Console.WriteLine(string.Format("Current thread {0} deal with {1}.", Thread.CurrentThread.ManagedThreadId, num));
- }
- }
- return;
- }
static private void ThreadWorker(object state)
{
int num = (int)state;
if (num > 0)
{
for (int i = 0; i < 3; i++)
{
Thread.Sleep(1000);// sleep for one second
num++;
Console.WriteLine(string.Format("Current thread {0} deal with {1}.", Thread.CurrentThread.ManagedThreadId, num));
}
}
return;
}
就是一个简单的隔一秒增加一的自增器。
调用ThreadPool的方法很简单,需要引用System.Threading。接着使用QueueUserWorkItem或者UnsafeQueueUserWorkerItem。
- public static class Th read Pool {
- }
- public static bool QueueUserWorkltem (WaitCa llback ca llBack) ;
- public stat ic bool QueueUserWork ltem (
- WaitCallback callBack,
- object state
- ) ;
- [Secu rityPermi ssion (Secu rityAction .Lin kDemand, Flags=
- Secu rityPermi ssionFlag . ControlPol icy l
- Secu rityPermi ssionF lag . Cont rolEvidence )]
- public stat ic bool UnsafeQueueUserWorkltem (
- WaitCallback callBac k,
- obj ect state
- ) ;
public static class Th read Pool {
}
public static bool QueueUserWorkltem (WaitCa llback ca llBack) ;
public stat ic bool QueueUserWork ltem (
WaitCallback callBack,
object state
) ;
[Secu rityPermi ssion (Secu rityAction .Lin kDemand, Flags=
Secu rityPermi ssionFlag . ControlPol icy l
Secu rityPermi ssionF lag . Cont rolEvidence )]
public stat ic bool UnsafeQueueUserWorkltem (
WaitCallback callBac k,
obj ect state
) ;
可以接受一个参数传递给异步线程。
结合前面的自增器,下面例子是申请5个异步线程来运行上述的自增器,传递的值是10,20,30,40,50。接着等待足够长的时间来等待所有线程池里的工作完成。
- for (int i = 1; i <= 5; i++)
- ThreadPool.QueueUserWorkItem(ThreadWorker, i * 10);
- for (int i = 0; i < 5; i++)
- Thread.Sleep(1000);// wait for all threads finish
for (int i = 1; i <= 5; i++)
ThreadPool.QueueUserWorkItem(ThreadWorker, i * 10);
for (int i = 0; i < 5; i++)
Thread.Sleep(1000);// wait for all threads finish
从上面可以看到线程池的使用非常的简单,只需要提供符合waitcallback的委托的工作函数,接着调用QueueUserWorkItem或者UnsafeQueueUserWorkItem添加到线程池里就ok了。但是线程池的缺点也是有的,首先,没有简单的方法来知道工作线程已经完成了,不想asynchronous中提供那么多种等待异步结束的方法,更谈不上能够添加回调函数,也不想thread那样可以使用join函数来等待线程或者使用abort函数中断正在运行的工作项。同时由于它是后台线程,也就是意味如果没有前台线程,程序就关闭,所有正在运行的或者还留在线程池里的工作项就给强制中断。所以不能将一些类似保存退出的操作放进线程池里,因为没有保证机制能够证明所有的线程池里正在运行或者还没运行的工作项能够具有明确的行为。
同步机制
ThreadPool没有提供简单的方法来获取工作线程已经结束,所以需要通过事件或者其他的内核对象来实现同步机制。
1. 如果需要获取某一个工作线程是否结束可以使用手动事件来实现。
- static private ManualResetEvent finish = new ManualResetEvent(false);
- static private void ThreadWorker(object state)
- {
- {
- for (int i = 0; i < 5; i++)
- {
- Thread.Sleep(1000);
- Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i + 1);
- }
- finish.Set();
- }
- return;
- }
static private ManualResetEvent finish = new ManualResetEvent(false);
static private void ThreadWorker(object state)
{ {
for (int i = 0; i < 5; i++)
{
Thread.Sleep(1000);
Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i + 1);
}
finish.Set();
}
return;
}
主调函数实现代码为:
- ThreadPool.QueueUserWorkItem(ThreadWorker);
- finish.WaitOne();
- finish.Close();
ThreadPool.QueueUserWorkItem(ThreadWorker);
finish.WaitOne();
finish.Close();
或者使用匿名委托函数加using关键字保证了事件的释放
- using (ManualResetEvent finish = new ManualResetEvent(false))
- {
- ThreadPool.QueueUserWorkItem(delegate {
- for (int i = 0; i < 5; i++)
- {
- Thread.Sleep(1000);
- Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i + 1);
- }
- finish.Set();
- });
- finish.WaitOne();
- }
using (ManualResetEvent finish = new ManualResetEvent(false))
{
ThreadPool.QueueUserWorkItem(delegate {
for (int i = 0; i < 5; i++)
{
Thread.Sleep(1000);
Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i + 1);
}
finish.Set();
});
finish.WaitOne();
}
2.如果需要获取多个工作线程是否结束可以通过最后一个工作线程来触发事件来实现。
如果使用前面的方法给每一个工作线程分配一个事件,然后等待所有事件给触发,是非常低效的,可以使用如下方法
- using (ManualResetEvent finish = new ManualResetEvent(false))
- {
- int nTask = 10;
- for (int i = 0; i < nTask; i++)
- {
- ThreadPool.QueueUserWorkItem(delegate
- {
- for (int j = 0; j < 3; j++)
- {
- Thread.Sleep(1000);
- Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i * 10 + j + 1);
- }
- if (Interlocked.Decrement(ref nTask) == 0)
- {
- finish.Set();
- }
- }, i);
- }
- finish.WaitOne();
- }
using (ManualResetEvent finish = new ManualResetEvent(false))
{
int nTask = 10;
for (int i = 0; i < nTask; i++)
{
ThreadPool.QueueUserWorkItem(delegate
{
for (int j = 0; j < 3; j++)
{
Thread.Sleep(1000);
Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i * 10 + j + 1);
}
if (Interlocked.Decrement(ref nTask) == 0)
{
finish.Set();
}
}, i);
}
finish.WaitOne();
}
不过注意由于使用过了匿名委托所以i的值是不确定,有可能是1或者是10,所以为了改变这种情况不能使用匿名委托,
- static private int _num = 10;
- static private ManualResetEvent finish = new ManualResetEvent(false);
- static private void ThreadWork(object state)
- {
- int i = (int)state;
- for (int j = 0; j < 3; j++)
- {
- Thread.Sleep(1000);
- Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i * 10 + j + 1);
- }
- if (Interlocked.Decrement(ref _num) == 0)
- finish.Set();
- }
static private int _num = 10;
static private ManualResetEvent finish = new ManualResetEvent(false);
static private void ThreadWork(object state)
{
int i = (int)state;
for (int j = 0; j < 3; j++)
{
Thread.Sleep(1000);
Console.WriteLine("current thead {0} wait for {1} seconds.", Thread.CurrentThread.ManagedThreadId, i * 10 + j + 1);
}
if (Interlocked.Decrement(ref _num) == 0)
finish.Set();
}
主调函数实现代码为:
- for (int i = 0; i < _num; i++)
- {
- ThreadPool.QueueUserWorkItem(ThreadWork, i);
- }
- finish.WaitOne();
- finish.Close();
C#多线程实现方法——线程池(Thread Pool)的更多相关文章
- 简易线程池Thread Pool
1. 基本思路 写了个简易的线程池,基本的思路是: 有1个调度线程,负责维护WorkItem队列.管理线程(是否要增加工作线程).调度(把工作项赋给工作线程)等 线程数量随WorkItem的量动态调整 ...
- MySQL线程池(THREAD POOL)的原理
MySQL常用(目前线上使用)的线程调度方式是one-thread-per-connection(每连接一个线程),server为每一个连接创建一个线程来服务,连接断开后,这个线程进入thread_c ...
- 线程池 (thread pool) 的类型与实现方式
在许多应用中需要频繁的创建许多生命周期很短的线程,如果用传统方法的话就会造成大量的资源了浪费,java的设计者们考虑到了这点在java中加入了线程池这个特性,它负责管理大量的线程的创建销毁等操作. 首 ...
- 使用boost实现线程池thread pool | boost thread pool example
本文首发于个人博客https://kezunlin.me/post/f241bd30/,欢迎阅读! boost thread pool example Guide boost thread pool ...
- Java多线程系列--“JUC线程池”06之 Callable和Future
概要 本章介绍线程池中的Callable和Future.Callable 和 Future 简介示例和源码分析(基于JDK1.7.0_40) 转载请注明出处:http://www.cnblogs.co ...
- Java多线程系列--“JUC线程池”01之 线程池架构
概要 前面分别介绍了"Java多线程基础"."JUC原子类"和"JUC锁".本章介绍JUC的最后一部分的内容——线程池.内容包括:线程池架构 ...
- Java多线程-新特性-线程池
Sun在Java5中,对Java线程的类库做了大量的扩展,其中线程池就是Java5的新特征之一,除了线程池之外,还有很多多线程相关的内容,为多线程的编程带来了极大便利.为了编写高效稳定可靠的多线程程序 ...
- 温故知新-java多线程&深入理解线程池
文章目录 摘要 java中的线程 java中的线程池 线程池技术 线程池的实现原理 简述 ThreadPoolExecutor是如何运行的? 线程池运行的状态和线程数量 任务执行机制 队列缓存 Wor ...
- Java多线程系列--“JUC线程池”02之 线程池原理(一)
概要 在上一章"Java多线程系列--“JUC线程池”01之 线程池架构"中,我们了解了线程池的架构.线程池的实现类是ThreadPoolExecutor类.本章,我们通过分析Th ...
随机推荐
- 【POJ】2236 Wireless Network
题目链接:http://poj.org/problem?id=2236 题意:给你n台计算机的坐标.d是可通信的最大距离.有两个操作. 1.O p 表示修复计算机p. 2.S p q表示询问pq是否能 ...
- 错误Exception in thread "main" java.lang.NoClassDefFoundError: org/apache/hadoop/fs/FSDataInputStream排查思路
spark1(默认CDH自带版本)不存在这个问题,主要是升级了spark2(CDHparcel升级)版本安装后需要依赖到spark1的旧配置去读取hadoop集群的依赖包. 1./etc/spark2 ...
- colormap 参数及对应色卡
[参考] [1]matlab帮助文档
- Perl 数据类型
Perl 数据类型 Perl 是一种弱类型语言,所以变量不需要指定类型,Perl 解释器会根据上下文自动选择匹配类型. Perl 有三个基本的数据类型:标量.数组.哈希.以下是这三种数据类型的说明: ...
- BZOJ 1037 (ZJOI 2008) 生日聚会
题目描述 今天是hidadz小朋友的生日,她邀请了许多朋友来参加她的生日party. hidadz带着朋友们来到花园中,打算坐成一排玩游戏.为了游戏不至于无聊,就座的方案应满足如下条件: 对于任意连续 ...
- hdu多校第四场 1003 (hdu6616) Divide the Stones 机智题
题意: 给你重量分别为1到n的n个石头,让你分成重量相等,数量也相等的k组,保证k是n的约数.问你能不能分配,如果能,输出具体的分配方案. 题解: 首先,如果1到n之和不能整除k,那么一定不能如题意分 ...
- error C2872: 'ULONG_PTR' : ambiguous symbol
转自VC错误:http://www.vcerror.com/?p=74 问题描述: 错误:error C2872: 'ULONG_PTR' : ambiguous symbol 解决方法: 详细的解决 ...
- 如何清除本机DNS缓存
如何清除本机DNS缓存 在实际应用过程中可能会遇到DNS解析错误的问题,就是说当我们访问一个域名时无法完成将其 解析到IP地址的工作,而直接输入网站IP却可以正常访问,这就是因为DNS解析出现故障造成 ...
- VS2010-MFC(对话框:字体对话框)
转自:http://www.jizhuomi.com/software/175.html 字体对话框的作用是用来选择字体.我们也经常能够见到.MFC使用CFontDialog类封装了字体对话框的所有操 ...
- laravel装饰者模式例子
interface Decorator{ public function display(); } class XiaoFang implements Decorator { private $nam ...