有时候我们进行很多的数据对比运算时,单线程的程序显得很慢,这个时候可以用多线程并发运行:
int maxThread = 10; //10个并发线程 int currTNum = 0; WaitHandle[] whs = new WaitHandle[maxThread]; //WaitHanle类MSDN上的定义: 封装了一些对象,这些对象等待着独占访问共享资源,这里相当于占用10个资源的意思 for (int i = 0; i < whs.Length; i++) { whs[i] = new AutoResetEvent(false); //AutoResetEvent是一个开关,设置为true时,whs占用资源对象就会自动启动. 设置为false时,只有进行 AutoResetEvent.set()之后才能启动 } int sortIdx = 0; //循环遍历一千个参数,对这1000个参数进行并发执行 Dotask()函数 for (int i= 0;i<1000;i++) { var ex = i; currTNum++; //线程数加一 if (currTNum >= maxThread) { //当前线程数大于或者等于最大线程数时 ,即10个线程数全部被占满,这时等待释放资源,把释放资源的线程编号给sortldx int freeIdx = WaitHandle.WaitAny(whs); currTNum--; sortIdx = freeIdx; } else { //10个线程未占满时,依次给编号0到9 sortIdx = currTNum - 1; } //线程池队列,依次执行线程 ThreadPool.QueueUserWorkItem(new WaitCallback((p) => { Dotask(p); //要执行的函数,这里的p等于ex的内容 (whs[sortIdx] as AutoResetEvent).Set(); }), ex); //唤醒线程 }
如果在for()循环外直接加 Messagebox.show("结束了"); 会发现 有时候已经弹出窗口了,但 仍有线程在运行。
这是因为for()循环虽然进行完了,但线程池中仍有线程在等待执行.
那我们什么时候才知道任务执行完了呢 这里我们要判断线程池中的线程是否执行完 执行完了 再弹出提示
int maxWorkerThreads, workerThreads; int portThreads; ThreadPool.GetMaxThreads(out maxWorkerThreads, out portThreads); ThreadPool.GetAvailableThreads(out workerThreads, out portThreads); if (maxWorkerThreads - workerThreads == 0) { Messagebox.show("结束了"); break; }