C#线程学习笔记三:线程池中的I/O线程
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/20/MultiThreads.html,记录一下学习过程以备后续查用。
一、I/O线程实现对文件的异步
1.1 I/O线程介绍:
对于线程所执行的任务来说,可以把线程分为两种类型:工作者线程和I/O线程。
工作者线程用来完成一些计算的任务,在任务执行的过程中,需要CPU不间断地处理,所以,在工作者线程的执行过程中,CPU和线程的资源是充分利用的。
I/O线程主要用来完成输入和输出的工作,在这种情况下, 计算机需要I/O设备完成输入和输出的任务。在处理过程中,CPU是不需要参与处理过程的,此时正在运行的线程
将处于等待状态,只有等任务完成后才会有事可做, 这样就造成线程资源浪费的问题。为了解决这样的问题,可以通过线程池来解决这样的问题,让线程池来管理线程。
对于I/O线程,我们可以将输入输出操作分成三个步骤:启动、实际输入输出、处理结果。用于实际输入输出可由硬件完成,并不需要CPU的参与,而启动和处理结果也可以
不在同一个线程上,这样就可以充分利用线程资源。在.Net中通过以Begin开头的方法来完成启动,以End开头的方法来处理结果,这两个方法可以运行在不同的线程,这样我们
就实现了异步编程了。
1.2 .Net中如何使用异步
注意:
其实当我们调用Begin开头的方法,就是将一个I/O线程排入到线程池中(由.Net机制帮我们实现)。
注:工作者线程由线程池管理,直接调用ThreadPool.QueueUserWorkItem方法来将工作者线程排入到线程池中。
在.NET Framework中的FCL中有许多类型能够对异步操作提供支持,其中在FileStream类中就提供了对文件的异步操作的方法。
FileStream类要调用I/O线程要实现异步操作,首先要建立一个FileStream对象,然后通过下面的构造函数来初始化FileStream对象实现异步操作(异步读取和异步写入):
public FileStream (string path, FileMode mode, FileAccess access, FileShare share,int bufferSize,bool useAsync)
其中path代表文件的相对路径或绝对路径,mode代表如何打开或创建文件,access代表访问文件的方式,share代表文件如何由进程共享,buffersize代表缓冲区的大小,
useAsync代表使用异步I/O还是同步I/O,设置为true时,表示使用异步I/O。
下面代码演示异步写入文件:
class Program
{
static void Main(string[] args)
{
#region I/O线程:异步写入文件
const int maxSize = ;
ThreadPool.SetMaxThreads(, );
PrintMessage("Main thread start."); //初始化FileStream对象
FileStream fileStream = new FileStream("Test.txt", FileMode.OpenOrCreate, FileAccess.ReadWrite, FileShare.ReadWrite, , true); //打印文件流打开的方式
Console.WriteLine("Filestream is {0}opened with asynchronously.", fileStream.IsAsync ? "" : "not "); byte[] writeBytes = new byte[maxSize];
string writeMessage = "An operation use asynchronous method to write message......";
writeBytes = Encoding.Unicode.GetBytes(writeMessage);
Console.WriteLine("Message sizes is:{0} bytes.\n", writeBytes.Length);
//调用异步写入方法将信息写入到文件中
fileStream.BeginWrite(writeBytes, , writeBytes.Length, new AsyncCallback(EndWriteCallback), fileStream);
fileStream.Flush();
Console.Read();
#endregion
} /// <summary>
/// 打印线程池信息
/// </summary>
/// <param name="data"></param>
private static void PrintMessage(string data)
{
//获得线程池中可用的工作者线程数量及I/O线程数量
ThreadPool.GetAvailableThreads(out int workThreadNumber, out int ioThreadNumber); Console.WriteLine("{0}\n CurrentThreadId is:{1}\n CurrentThread is background:{2}\n WorkerThreadNumber is:{3}\n IOThreadNumbers is:{4}\n",
data,
Thread.CurrentThread.ManagedThreadId,
Thread.CurrentThread.IsBackground.ToString(),
workThreadNumber.ToString(),
ioThreadNumber.ToString());
} /// <summary>
/// 当数据写入文件完成后调用此方法来结束异步写操作
/// </summary>
/// <param name="asyncResult"></param>
private static void EndWriteCallback(IAsyncResult asyncResult)
{
Thread.Sleep();
PrintMessage("Asynchronous method start."); FileStream filestream = asyncResult.AsyncState as FileStream; //结束异步写入数据
filestream.EndWrite(asyncResult);
filestream.Close();
}
}
运行结果如下:
从运行结果可以看出,此时是调用线程池中的I/O线程去执行回调函数的,同时在项目的bin\Debug文件目录下生成了一个Test.txt文件。
下面代码演示异步读取文件:
class Program
{
//异步读取文件
const int maxSize = ;
private static readonly byte[] readBytes = new byte[maxSize]; static void Main(string[] args)
{
#region I/O线程:异步读取文件
ThreadPool.SetMaxThreads(, );
PrintMessage("Main thread start."); // 初始化FileStream对象
FileStream fileStream = new FileStream("Test.txt", FileMode.OpenOrCreate, FileAccess.ReadWrite, FileShare.ReadWrite, , false); // 异步读取文件内容
fileStream.BeginRead(readBytes, , readBytes.Length, new AsyncCallback(EndReadCallback), fileStream);
Console.Read();
#endregion
} /// <summary>
/// 打印线程池信息
/// </summary>
/// <param name="data"></param>
private static void PrintMessage(string data)
{
//获得线程池中可用的工作者线程数量及I/O线程数量
ThreadPool.GetAvailableThreads(out int workThreadNumber, out int ioThreadNumber); Console.WriteLine("{0}\n CurrentThreadId is:{1}\n CurrentThread is background:{2}\n WorkerThreadNumber is:{3}\n IOThreadNumbers is:{4}\n",
data,
Thread.CurrentThread.ManagedThreadId,
Thread.CurrentThread.IsBackground.ToString(),
workThreadNumber.ToString(),
ioThreadNumber.ToString());
} /// <summary>
/// 当数据读取文件完成后调用此方法来结束异步写操作
/// </summary>
/// <param name="asyncResult"></param>
private static void EndReadCallback(IAsyncResult asyncResult)
{
Thread.Sleep();
PrintMessage("Asynchronous method start."); // 把AsyncResult.AsyncState转换为State对象
FileStream readStream = (FileStream)asyncResult.AsyncState;
int readLength = readStream.EndRead(asyncResult);
if (readLength <= )
{
Console.WriteLine("Read error.");
return;
} string readMessage = Encoding.Unicode.GetString(readBytes, , readLength);
Console.WriteLine("Read message is :" + readMessage);
readStream.Close();
}
}
运行结果如下:
二、I/O线程实现对请求的异步
我们同样可以利用I/O线程来模拟浏览器对服务器请求的异步操作,在.NET类库中的WebRequest类提供了异步请求的支持。
下面代码演示异步请求:
class Program
{
static void Main(string[] args)
{
#region I/O线程:异步请求
ThreadPool.SetMaxThreads(, );
PrintMessage("Main thread start."); // 发出一个异步Web请求
WebRequest webrequest = WebRequest.Create("https://www.cnblogs.com/");
webrequest.BeginGetResponse(ProcessWebResponse, webrequest); Console.Read();
#endregion
} /// <summary>
/// 打印线程池信息
/// </summary>
/// <param name="data"></param>
private static void PrintMessage(string data)
{
//获得线程池中可用的工作者线程数量及I/O线程数量
ThreadPool.GetAvailableThreads(out int workThreadNumber, out int ioThreadNumber); Console.WriteLine("{0}\n CurrentThreadId is:{1}\n CurrentThread is background:{2}\n WorkerThreadNumber is:{3}\n IOThreadNumbers is:{4}\n",
data,
Thread.CurrentThread.ManagedThreadId,
Thread.CurrentThread.IsBackground.ToString(),
workThreadNumber.ToString(),
ioThreadNumber.ToString());
} /// <summary>
/// Web请求回调函数
/// </summary>
/// <param name="result"></param>
private static void ProcessWebResponse(IAsyncResult result)
{
Thread.Sleep();
PrintMessage("Asynchronous method start."); WebRequest webRequest = (WebRequest)result.AsyncState;
using (WebResponse wr = webRequest.EndGetResponse(result))
{
Console.WriteLine("Content length is : " + wr.ContentLength);
}
}
}
运行结果如下:
C#线程学习笔记三:线程池中的I/O线程的更多相关文章
- python 学习笔记(三)根据字典中值的大小对字典中的项排序
字典的元素是成键值对出现的,直接对字典使用sorted() 排序,它是根据字典的键的ASCII编码顺序进行排序,要想让字典根据值的大小来排序,可以有两种方法来实现: 一.利用zip函数将字典数据转化为 ...
- Linux进程线程学习笔记:运行新程序
Linux进程线程学习笔记:运行新程序 周银辉 在上一篇中我们说到,当启动一个新进程以后,新进程会复制父进程的大部份上下 ...
- C#线程学习笔记二:线程池中的工作者线程
本笔记摘抄自:https://www.cnblogs.com/zhili/archive/2012/07/18/ThreadPool.html,记录一下学习过程以备后续查用. 一.线程池基础 首先,创 ...
- 转:专题三线程池中的I/O线程
上一篇文章主要介绍了如何利用线程池中的工作者线程来实现多线程,使多个线程可以并发地工作,从而高效率地使用系统资源.在这篇文章中将介绍如何用线程池中的I/O线程来执行I/O操作,希望对大家有所帮助. 目 ...
- C#线程学习笔记十:async & await入门三
一.Task.Yield Task.Yield简单来说就是创建时就已经完成的Task,或者说执行时间为0的Task,或者说是空任务,也就是在创建时就将Task的IsCompeted值设置为0. 我们知 ...
- C#线程学习笔记九:async & await入门二
一.异步方法返回类型 只能返回3种类型(void.Task和Task<T>). 1.1.void返回类型:调用方法执行异步方法,但又不需要做进一步的交互. class Program { ...
- C#线程学习笔记七:Task详细用法
一.Task类简介: Task类是在.NET Framework 4.0中提供的新功能,主要用于异步操作的控制.它比Thread和ThreadPool提供了更为强大的功能,并且更方便使用. Task和 ...
- java之jvm学习笔记三(Class文件检验器)
java之jvm学习笔记三(Class文件检验器) 前面的学习我们知道了class文件被类装载器所装载,但是在装载class文件之前或之后,class文件实际上还需要被校验,这就是今天的学习主题,cl ...
- 学习笔记(三)--->《Java 8编程官方参考教程(第9版).pdf》:第十章到十二章学习笔记
回到顶部 注:本文声明事项. 本博文整理者:刘军 本博文出自于: <Java8 编程官方参考教程>一书 声明:1:转载请标注出处.本文不得作为商业活动.若有违本之,则本人不负法律责任.违法 ...
随机推荐
- 在Windows Server 2019通过Docker Compose部署Asp.Net Core
一.安装Docker Enterprise 安装文档是: https://docs.docker.com/install/windows/docker-ee/ 安装完成后,如下图 二.首先,拉取一个W ...
- 面向对象之classmethod和staticmethod(python内置装饰器)
对象的绑定方法复习classmethodstaticmethod TOC 对象的绑定方法复习 由对象来调用 会将对象当做第一个参数传入 若对象的绑定方法中还有其他参数,会一并传入 classmetho ...
- 【2018寒假集训 Day1】【位运算】桐桐的运输方案
桐桐的运输方案(transp) [问题描述] 桐桐有 N 件货物需要运送到目的地,它们的重量和价值分别记为: 重量:W1,W2,…,Wn: 价值:V1,V2,…,Vn: 已知某辆货车的最大载货量为 X ...
- vue 做一个简单的TodoList
目录结构 index.html <!DOCTYPE html> <html> <head> <meta charset="utf-8"&g ...
- CentOS 7 Cobbler 自动化安装系统
在上一篇Cobbler 安装中,配置好了Cobbler,下面来配置自动化安装 配置cobbler-DHCP # 修改settings中参数,由cobbler控制dhcp [root@cobbler ~ ...
- python calendar 时间处理类库
#python中的calendar import calendar #返回指定年的某月 def get_month(year, month): return calendar.month(year, ...
- 限定某个目录禁止解析php、限制user_agent、php相关配置
6月1日任务 11.28 限定某个目录禁止解析php11.29 限制user_agent11.30/11.31 php相关配置扩展apache开启压缩 http://ask.apelearn.com/ ...
- windows下安装python numpy+scipy+matlotlib+scikit-learn等流行库
(1)请不要直接使用 pip install scikit-learn pip install Numpy pip install Scipy pip install Matplotlib 命令安 ...
- 重启testjenkins的步骤
在linux下编译caffe的过程中,发生错误,导致linux系统蹦了,没办法,重启linux系统. 之前安装在docker下的jenkins也停掉了. 先启动jenkins的步骤如下: 1.先启动d ...
- 关于iframe/子窗体与父窗体的交互
父子窗体交互方式 通过contentWindow交互 通过postMessage交互 通过contentWindow交互 主窗体内嵌的iframe或者是其通过js打开的新窗口都可以通过contentW ...