线程池(摘自C#高级编程第7版)
程序员文章站
2023-11-14 21:19:10
1、需求背景 创建线程需要时间。如果有不同的小任务完成,就可以事先创建许多线程,在应完成这些任务时发出请求。这个线程数最好在需要更多的线程时增加,在需要释放资源时减少。 2、线程池出场 不需要自己创建这样一个列表。该列表由ThreadPool类托管。这个类会在需要时增减池中线程的线程数,直到最大的线 ......
1、需求背景
创建线程需要时间。如果有不同的小任务完成,就可以事先创建许多线程,在应完成这些任务时发出请求。这个线程数最好在需要更多的线程时增加,在需要释放资源时减少。
2、线程池出场
不需要自己创建这样一个列表。该列表由threadpool类托管。这个类会在需要时增减池中线程的线程数,直到最大的线程数。池中的最大线程数是可配置的。
在双核cpu中,默认设置为1023个工作线程和1000个i/o线程。四核cpu中,默认设置为2047个工作线程和1000个i/o线程。
也可以指定在创建线程池时应立即启动的最小线程数,以及线程池中可用的最大线程数。
如果有更多的作业要处理,线程池中线程的个数也到了极限,最新的作业就要排队,且必须等待线程完成其任务。
3、举个栗子
下面的实例首先读取工作线程和i/o线程的最大线程数,把这些信息写入控制台中,接着在for循环中,调用threadpool.queueuserworkitem()方法,传递一个waitcallback类型的委托,把jobforathread()方法赋予线程池中的线程。线程池接收到这个请求后,就会从池中选择一个线程,来调用该方法。
如果线程池还没有运行,就会创建一个线程池,并启动第一个线程。如果线程池已经在运行,且有一个空闲线程来完成该任务,就把该任务传递给这个线程。
class clrthread2 { public static void threadmethod() { int nworkthreads; int ncompletionportthreads; threadpool.getmaxthreads(out nworkthreads, out ncompletionportthreads); console.writeline("max worker threads:{0},i/o completion threads:{1}", nworkthreads, ncompletionportthreads); for (int i = 0; i < 5; i++) { threadpool.queueuserworkitem(jobforathread); } thread.sleep(3000); } static void jobforathread(object state) { for (int i = 0; i < 3; i++) { console.writeline("loop {0},running inside pooled thread {1}", i, thread.currentthread.managedthreadid); thread.sleep(50); } } }
4、线程池的使用限制
线程池用起来简单,但它有一些限制:
- 线程池中的所有线程都是后台线程。如果进程的所有前台线程都结束了,所有的后台线程就会停止。不能把入池的线程改为前台线程。
- 不能给入池的线程设置优先级或名称
- 对于com对象,入池的所有线程都是多线程单元(multitihreaded apartment,mta)线程。许多com对象都需要单线程单元(single-threaded apartmeng,sta)线程
- 入池的线程只能用于时间较短的任务。如果线程要一直运行(如word的拼写检查器线程),就应该使用thread类创建一个线程。