C#:多线程(Smart Thread Pool)-1

本文详细介绍了C#实现的智能线程池,具有动态调整线程数量、返回值、取消未开始任务、调用线程上下文等功能。通过代码示例展示了如何捕获异常、合作取消任务和获取异常信息,强调了线程池在异常处理和资源管理方面的高效特性。
摘要由CSDN通过智能技术生成

原文来自Smart Thread Pool,这篇文章很长(感觉不是一篇文章,而是一本书),文章描叙为:A .Net Thread Pool fully implement in C# with many features。

文中作者整合/改进了别人的线程池,变成我们现在看到的智能线程池(Smart Thread Pool,类库版本.Net 2.0)。它具有以下特征:

1)线程池中线程的数量根据工作量的变化而改变;

2)Work Items可以返回一个值;

3)如果Work Item没有开始执行,那它可以被取消的;

4)当work Item被执行时,调用线程的上下文(context)可以被使用的(不是很明白?);

5)使用最小数量的Window32事件句柄,可以保证在应用中句柄的数量不会大幅扩大(也就是使用系统资源比较少);

6)调用者可以等待多个或所有的WorkItem完成(这个很有用的);

7)一个Work Item可以有PostExecute的回掉函数,它主要用于当Work Item完成后开始被执行;

8)伴随Work Item的状态对象(state Object)可以被自动释放;

9)Work Item运行过程中的异常是被返回到调用者;

10)Work Item是有优先级的;

11)Work Item有组的区分;

12)调用者可以暂停线程池和Work Item组的开始;

13)线程也有优先级;

14)线程有初始化及结束事件;

15)支持winCE平台(没玩过);

16)支持Action<T>和Func<T>泛型方法(也不是很明白);

17)<t>SilverLight(Web前端应用开发)也支持;

18)Mono(.Net平台下混合编程语言)也支持;

19)<t>Performance Counters (Windows and internal);

20)WorkItem支持超时(主动);

21)线程ApartmentState;

22)线程是否为后台线程;

23)线程名字模板;

24)支持Windowns Phone(受限);

25)可以设定线程的MaxStackSize;


接下来我们通过实例代码来熟悉这个智能线程池。为了简化,通过控制台来直接调用。

CatchException:
public class CatchExceptionExample
{
        public void DoWork()
        {
            SmartThreadPool smartThreadPool = new SmartThreadPool();

            //the 3rd param double is the result
            IWorkItemResult<double> wir = smartThreadPool.QueueWorkItem(new Func<double, double, double>(DoDiv), 10.0, 0.0);

            try
            {
                double result = wir.Result;
                Console.WriteLine(result.ToString());
            }
            // Catch the exception that Result threw
            catch (WorkItemResultException e)
            {
                // Dump the inner exception which DoDiv threw
                Debug.WriteLine(e.InnerException);
            }

            smartThreadPool.Shutdown();
        }

        private double DoDiv(double x, double y)
        {
            return x / y;
        }
}

控制台输出的结果:


这印证了上面的第九条:Work Item返回的异常会返回到调用者。实际调用时是这样表达的:

public static void Main()
{
      CatchExceptionExample catchExc1 = new CatchExceptionExample();
      catchExc1.DoWork();
      Console.ReadKey();
}
CooperativeCancel:
public class CooperativeCancelExample
{
	public void DoWork(object state) 
	{ 
	    SmartThreadPool smartThreadPool = new SmartThreadPool();

            // Queue the work item
	    IWorkItemResult wir = smartThreadPool.QueueWorkItem(DoRealWork); 

            // Give the work item some time to complete.
            Thread.Sleep(1000);

            // If the work item hasn't completed yet then cancel it.
            if (!wir.IsCompleted)
            {
                Console.WriteLine("Work item canceling started");
                wir.Cancel();
                Console.WriteLine("Work item canceling ended");
            }

	    smartThreadPool.Shutdown();
	} 

	// Do some lengthy work
        private void DoRealWork()
        {
            Console.WriteLine("In DoRealWork()");

            Random rm = new Random();
            Thread.Sleep(1);
            int time = rm.Next(10, 30);

            // Do something here.
            Thread.Sleep(time);
            Console.WriteLine("Sleep before do real job,time=" + time.ToString());

            // Sample SmartThreadPool.IsWorkItemCanceled
            if (SmartThreadPool.IsWorkItemCanceled)
            {
                Console.WriteLine("Work Item canceled, return directly");
                return;
            }

            // Sample the SmartThreadPool.IsWorkItemCanceled in a loop
            while (!SmartThreadPool.IsWorkItemCanceled)
            {
                // Do some real work here
                Thread.Sleep(1);
                time = rm.Next(1000, 3000);
                Thread.Sleep(time);
                Console.WriteLine("Work Item do real job, time=" + time.ToString());
            }
        }
}

控制台输出的结果:


结果显示:已经进入到DoRealWork()中,由于没有结束,所以可以取消,对应上面的第三项。

GetException:
public class GetExceptionExample
{
	private class DivArgs
	{
		public int x;
		public int y;
	}

	public void DoWork() 
	{ 
		SmartThreadPool smartThreadPool = new SmartThreadPool();

		DivArgs divArgs = new DivArgs();
		divArgs.x = 10;
		divArgs.y = 0;

		IWorkItemResult wir = smartThreadPool.QueueWorkItem(new WorkItemCallback(this.DoDiv), divArgs);

		Exception e;
		object obj = wir.GetResult(out e);
		// e contains the expetion that DoDiv threw
		if(null == e)
		{
			int result = (int)obj;
                        Console.WriteLine("No exception, got result= " + result.ToString());
		}
		else
		{
                        // Do something with the exception
                        Console.WriteLine("Got exception in result: " + e.ToString());
		}

		smartThreadPool.Shutdown();
	} 

	private object DoDiv(object state)
	{ 
		DivArgs divArgs = (DivArgs)state;
		return (divArgs.x / divArgs.y);
	}
}

控制台输出结果:


同样对应上面的第九项,对线程池中的结果进行检查,Exception也包含在结果中,区分异常e的方式是通过判断其是否为空。


SmartThreadPool是大名鼎鼎的.Net线程池项目,基于.Net开发,比.Net内置的线程池更胜一筹。1、为什么需要使用线程池(Thread Pool)减少线程间上下文切换。线程执行一定的时间片后,系统会自动把cpu切换给另一个线程使用,这时还需要保存当 前的线程上下文状态,并加载新线程的上下文状态。当程序中有大量的线程时,每个线程分得的时间片会越来越少,可能会出现线程未处理多少操作,就需要切换到 另一线程,这样频繁的线程间上下文切换会花费大量的cpu时间。减少内存占用。系统每创建一条物理线程,需要大概花费1MB的内存空间,许多程序喜欢先创建多条物理线程,并 周期轮询来处理各自的任务,这样既消耗了线程上下文切换的时间,还浪费了内存。这些任务可能只需要一条线程就能满足要求。假如某一任务需要执行较长的周 期,线程池还可以自动增加线程,并在空闲时,销毁线程,释放占用的内存。2、为什么不使用.Net默认的线程池.Net默认的线程池(ThreadPool)是一个静态类,所以是没办法自己创建一个新的程序池的。默认的线程池与应用程序域 (AppDomain)挂钩,一个AppDomain只有一个线程池。假如在线程池中执行了一个周期较长的任务,一直占用着其中一个线程,可能就会影响到 应用程序域中的其他程序的性能。例如,假如在Asp.Net的线程池中执行一个周期较长的任务,就会影响请求的并发处理能力(线程池默认有个最大线程 数)。 3、SmartThreadPool特性和优点    SmartThreadPool特性如下:可创建线程池实例。可动态调整线程池工作线程数量。WorkItem 可以返回信息。未执行 WorkItem 可被取消。WorkItem 执行时可使用调用者上下文。调用者可等待多个或全部 WorkItem 执行结束。WorkItem 允许拥有一个执行结束时被执行的 PostExecute 回调委托。可以向 WorkItem 传递一个状态对象,并且会在执行结束时自动调用 IDisposable.Dispose()。WorkItem 异常会传递给调用者。支持 WorkItem 分组。可挂起线程池或分组。可以设置 WorkItem 优先级。可以设置线程优先级。4、使用示例 最简单的使用方法:// 创建一个线程池 SmartThreadPool smartThreadPool = new SmartThreadPool();    // 执行任务 smartThreadPool.QueueWorkItem(() => {      Console.WriteLine("Hello World!"); });带返回值的任务:// 创建一个线程池 SmartThreadPool smartThreadPool = new SmartThreadPool();   // 执行任务 var result = smartThreadPool.QueueWorkItem(() => {     var sum = 0;     for (var i = 0; i  {     //模拟计算较长时间     Thread.Sleep(5000);       return 3; });   var result2 = smartThreadPool.QueueWorkItem(() => {     //模拟计算较长时间     Thread.Sleep(3000);       return 5; });   bool success = SmartThreadPool.WaitAll(     new IWorkItemResult[] { result1, result2 });   if (success) {     // 输出结果     Console.WriteLine(result1.Result);     Console.WriteLine(result2.Result); }5、结论 使用SmartThreadPool可以简单就实现支持多线程的程序,由线程池来管理线程,可以减少死锁的出现。SmartThreadPool还支持简单的生产者-消费者模式,当不需要对任务进行持久化时,还是很好用的。 6、扩展阅读 http://www.codeproject.com/KB/threads/smartthreadpool.aspx http://smartthreadpool.codeplex.com/http://www.albahari.com/threading/ 标签:线程池
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值