C#多线程之线程池篇2
在上一篇C#多线程之线程池篇1中,我们主要学习了如何在线程池中调用委托以及如何在线程池中执行异步操作,在这篇中,我们将学习线程池和并行度、实现取消选项的相关知识。
三、线程池和并行度
在这一小节中,我们将学习对于大量的异步操作,使用线程池和分别使用单独的线程在性能上有什么差异性。具体操作步骤如下:
1、使用Visual Studio 2015创建一个新的控制台应用程序。
2、双击打开“Program.cs”文件,编写代码如下所示:
using System;
using System.Diagnostics;
using System.Threading;
using static System.Console;
using static System.Threading.Thread; namespace Recipe03
{
class Program
{
static void UseThreads(int numberOfOperations)
{
using(var countdown=new CountdownEvent(numberOfOperations))
{
WriteLine("Scheduling work by creating threads");
for(int i = ; i < numberOfOperations; i++)
{
var thread = new Thread(() =>
{
Write($"{CurrentThread.ManagedThreadId},");
Sleep();
countdown.Signal();
});
thread.Start();
}
countdown.Wait();
WriteLine();
}
} static void UseThreadPool(int numberOfOperations)
{
using(var countdown=new CountdownEvent(numberOfOperations))
{
WriteLine("Starting work on a threadpool");
for(int i = ; i < numberOfOperations; i++)
{
ThreadPool.QueueUserWorkItem(_ =>
{
Write($"{CurrentThread.ManagedThreadId},");
Sleep();
countdown.Signal();
});
}
countdown.Wait();
WriteLine();
}
} static void Main(string[] args)
{
const int numberOfOperations = ;
var sw = new Stopwatch();
sw.Start();
UseThreads(numberOfOperations);
sw.Stop();
WriteLine($"Execution time using threads: {sw.ElapsedMilliseconds}"); sw.Reset();
sw.Start();
UseThreadPool(numberOfOperations);
sw.Stop();
WriteLine($"Execution time using the thread pool: {sw.ElapsedMilliseconds}");
}
}
}
3、运行该控制台应用程序,运行效果(每次运行效果可能不同)如下图所示:
在上述代码中,我们首先创建了500个线程来执行异步操作,我们发现使用每个单独的线程执行异步操作所消耗的时间为175毫秒。然后我们使用线程池来执行500个异步操作,我们发现所消耗的时间为9432毫秒。这说明使用线程池来执行大并发的异步操作会节省操作系统的内存和线程数,但是会严重影响应用程序的性能。
四、实现取消选项
在这一小节中,我们将学习如何在线程池中取消一个异步操作。具体步骤如下所示:
1、使用Visual Studio 2015创建一个新的控制台应用程序。
2、双击打开“Program.cs”文件,编写代码如下所示:
using System;
using System.Threading;
using static System.Console;
using static System.Threading.Thread; namespace Recipe04
{
class Program
{
// CancellationToken:传播有关应取消操作的通知。
static void AsyncOperation1(CancellationToken token)
{
WriteLine("Starting the first task");
for (int i = ; i < ; i++)
{
// IsCancellationRequested:获取是否已请求取消此标记。
// 如果已请求取消此标记,则为 true,否则为 false。
if (token.IsCancellationRequested)
{
WriteLine("The first task has been canceled.");
return;
}
Sleep(TimeSpan.FromSeconds());
}
WriteLine("The first task has completed succesfully");
} static void AsyncOperation2(CancellationToken token)
{
try
{
WriteLine("Starting the second task");
for (int i = ; i < ; i++)
{
// 如果已请求取消此标记,则引发 System.OperationCanceledException。
token.ThrowIfCancellationRequested();
Sleep(TimeSpan.FromSeconds());
}
WriteLine("The second task has completed succesfully");
}
catch (OperationCanceledException)
{
WriteLine("The second task has been canceled.");
}
} static void AsyncOperation3(CancellationToken token)
{
bool cancellationFlag = false;
// 注册一个将在取消此 System.Threading.CancellationToken 时调用的委托。
// Register的参数是一个Action类型的委托,该委托在取消 System.Threading.CancellationToken 时执行
token.Register(() => cancellationFlag = true);
WriteLine("Starting the third task");
for (int i = ; i < ; i++)
{
if (cancellationFlag)
{
WriteLine("The third task has been canceled.");
return;
}
Sleep(TimeSpan.FromSeconds());
}
WriteLine("The third task has completed succesfully");
} static void Main(string[] args)
{
// CancellationTokenSource:通知 System.Threading.CancellationToken,告知其应被取消。
using (var cts = new CancellationTokenSource())
{
// 获取与此 System.Threading.CancellationTokenSource 关联的 System.Threading.CancellationToken。
CancellationToken token = cts.Token;
ThreadPool.QueueUserWorkItem(_ => AsyncOperation1(token));
Sleep(TimeSpan.FromSeconds());
// 传达取消请求。
cts.Cancel();
} // CancellationTokenSource:通知 System.Threading.CancellationToken,告知其应被取消。
using (var cts = new CancellationTokenSource())
{
// 获取与此 System.Threading.CancellationTokenSource 关联的 System.Threading.CancellationToken。
CancellationToken token = cts.Token;
ThreadPool.QueueUserWorkItem(_ => AsyncOperation2(token));
Sleep(TimeSpan.FromSeconds());
// 传达取消请求。
cts.Cancel();
} // CancellationTokenSource:通知 System.Threading.CancellationToken,告知其应被取消。
using (var cts = new CancellationTokenSource())
{
// 获取与此 System.Threading.CancellationTokenSource 关联的 System.Threading.CancellationToken。
CancellationToken token = cts.Token;
ThreadPool.QueueUserWorkItem(_ => AsyncOperation3(token));
Sleep(TimeSpan.FromSeconds());
// 传达取消请求。
cts.Cancel();
} Sleep(TimeSpan.FromSeconds());
}
}
}
3、运行该控制台应用程序,运行效果如下图所示:
在上述代码中,我们使用了CancellationTokenSource和CancellationToken类,这两个类在.NET 4.0引入,现在已经成为取消异步操作事实上的标准。
在“AsyncOperation1”方法中,我们仅仅是轮询检查“CancellationToken.IsCancellationRequested”属性,如果该属性为true,这意味着我们的操作已被取消,我们必须放弃此次操作。
在“AsyncOperation2”方法中,我们调用CancellationToken的“ThrowIfCancellationRequested”方法来检查操作是否已被取消,如果已被取消,该方法会抛出OperationCanceledException异常,我们使用try/catch块捕获这个异常来中止异步操作的执行。
在“AsyncOperation3”方法中,我们调用CancellationToken的“Register”方法来注册一个异步操作被取消时被调用的回调方法。这种方式可以允许我们将取消操作的逻辑链接到另一个异步操作中。
C#多线程之线程池篇2的更多相关文章
- C#多线程之线程池篇3
在上一篇C#多线程之线程池篇2中,我们主要学习了线程池和并行度以及如何实现取消选项的相关知识.在这一篇中,我们主要学习如何使用等待句柄和超时.使用计时器和使用BackgroundWorker组件的相关 ...
- C#多线程之线程池篇1
在C#多线程之线程池篇中,我们将学习多线程访问共享资源的一些通用的技术,我们将学习到以下知识点: 在线程池中调用委托 在线程池中执行异步操作 线程池和并行度 实现取消选项 使用等待句柄和超时 使用计时 ...
- JUC源码分析-线程池篇(三)ScheduledThreadPoolExecutor
JUC源码分析-线程池篇(三)ScheduledThreadPoolExecutor ScheduledThreadPoolExecutor 继承自 ThreadPoolExecutor.它主要用来在 ...
- Qt中的多线程与线程池浅析+实例
1. Qt中的多线程与线程池 今天学习了Qt中的多线程和线程池,特写这篇博客来记录一下 2. 多线程 2.1 线程类 QThread Qt 中提供了一个线程类,通过这个类就可以创建子线程了,Qt 中一 ...
- Java 多线程:线程池
Java 多线程:线程池 作者:Grey 原文地址: 博客园:Java 多线程:线程池 CSDN:Java 多线程:线程池 工作原理 线程池内部是通过队列结合线程实现的,当我们利用线程池执行任务时: ...
- C#多线程之线程同步篇3
在上一篇C#多线程之线程同步篇2中,我们主要学习了AutoResetEvent构造.ManualResetEventSlim构造和CountdownEvent构造,在这一篇中,我们将学习Barrier ...
- C#多线程之线程同步篇2
在上一篇C#多线程之线程同步篇1中,我们主要学习了执行基本的原子操作.使用Mutex构造以及SemaphoreSlim构造,在这一篇中我们主要学习如何使用AutoResetEvent构造.Manual ...
- 重新想象 Windows 8 Store Apps (42) - 多线程之线程池: 延迟执行, 周期执行, 在线程池中找一个线程去执行指定的方法
[源码下载] 重新想象 Windows 8 Store Apps (42) - 多线程之线程池: 延迟执行, 周期执行, 在线程池中找一个线程去执行指定的方法 作者:webabcd 介绍重新想象 Wi ...
- ExecutorService 建立一个多线程的线程池的步骤
ExecutorService 建立一个多线程的线程池的步骤: 线程池的作用: 线程池功能是限制在系统中运行的线程数. 依据系统的环境情况,能够自己主动或手动设置线程数量.达到执行的最佳效果:少了浪费 ...
随机推荐
- 配置android sdk 环境
1:下载adnroid sdk安装包 官方下载地址无法打开,没有vpn,使用下面这个地址下载,地址:http://www.android-studio.org/
- 使用ServiceStack构建Web服务
提到构建WebService服务,大家肯定第一个想到的是使用WCF,因为简单快捷嘛.首先要说明的是,本人对WCF不太了解,但是想快速建立一个WebService,于是看到了MSDN上的这一篇文章 Bu ...
- 水印第三版 ~ 变态水印(这次用Magick.NET来实现,附需求分析和源码)
技能 汇总:http://www.cnblogs.com/dunitian/p/4822808.html#skill 以前的水印,只是简单走起,用的是原生态的方法.现在各种变态水印,于是就不再用原生态 ...
- Python的单元测试(二)
title: Python的单元测试(二) date: 2015-03-04 19:08:20 categories: Python tags: [Python,单元测试] --- 在Python的单 ...
- 几个有趣的WEB设备API 前端提高B格必备(一)——电池状态&震动api
受到同事启发,突然发现了几个有趣又实用的web api,没想到前端还有这么多有趣的东西可以玩~~简直过分. 1.电池状态API navigator.getBattery():这个api返回的是一个pr ...
- Windows下Visual studio 2013 编译 Audacity
编译的Audacity版本为2.1.2,由于实在windows下编译,其源代码可以从Github上取得 git clone https://github.com/audacity/audacity. ...
- Django admin定制化,User字段扩展[原创]
前言 参考上篇博文,我们利用了OneToOneField的方式使用了django自带的user,http://www.cnblogs.com/caseast/p/5909248.html , 但这么用 ...
- margin折叠-从子元素margin-top影响父元素引出的问题
正在做一个手机端电商项目,顶部导航栈的布局是一个div包含一个子div,如果给在正常文档流中的子div一个垂直margin-top,神奇的现象出现了,两父子元素的边距没变,但父div跟着一起往下走了! ...
- git命令行操作
从本地上传代码到仓库(假设已经建好仓库): 1.初始化: git init 2.将所有文件加入缓存区: git add * 3.提交当前工作空间的修改内容: git commit -m 'commit ...
- 解决maven下载jar慢的问题(如何更换Maven下载源)
修改 配置文件 maven 安装 路径 F:\apache-maven-3.3.9\conf 修改 settings.xml 在 <mirrors> <!-- mirror | Sp ...