国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > C# > 正文

C#線程處理系列之線程池中的I/O線程

2019-10-29 21:30:17
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了C#線程處理系列之線程池中的I/O線程,在這篇文章中將介紹如何用線程池中的I/O線程來執行I/O操作,感興趣的小伙伴們可以參考一下
 

一、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線程排入到線程池中(調用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.

下面通過代碼來學習下異步寫入文件:

using System;using System.IO;using System.Text;using System.Threading;namespace AsyncFile{  class Program  {    static void Main(string[] args)    {      const int maxsize = 100000;      ThreadPool.SetMaxThreads(1000,1000);      PrintMessage("Main Thread start");      // 初始化FileStream對象      FileStream filestream = new FileStream("test.txt", FileMode.OpenOrCreate, FileAccess.ReadWrite, FileShare.ReadWrite, 100, true);            //打印文件流打開的方式      Console.WriteLine("filestream is {0} opened 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 size is: {0} byte/n", writebytes.Length);      // 調用異步寫入方法比信息寫入到文件中      filestream.BeginWrite(writebytes, 0, writebytes.Length, new AsyncCallback(EndWriteCallback), filestream);      filestream.Flush();      Console.Read();    }    // 當把數據寫入文件完成后調用此方法來結束異步寫操作    private static void EndWriteCallback(IAsyncResult asyncResult)    {      Thread.Sleep(500);      PrintMessage("Asynchronous Method start");      FileStream filestream = asyncResult.AsyncState as FileStream;      // 結束異步寫入數據      filestream.EndWrite(asyncResult);      filestream.Close();    }    // 打印線程池信息    private static void PrintMessage(String data)    {      int workthreadnumber;      int iothreadnumber;      // 獲得線程池中可用的線程,把獲得的可用工作者線程數量賦給workthreadnumber變量      // 獲得的可用I/O線程數量給iothreadnumber變量      ThreadPool.GetAvailableThreads(out workthreadnumber, out 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());    }  }}

運行結果:

C#,線程處理,線程池,I/O線程

從運行結果可以看出,此時是調用線程池中的I/O線程去執行回調函數的,同時在工程所的的bin/Debug文件目錄下有生成一個text.txt文件,打開文件可以知道里面的內容正是你寫入的。

下面演示如何從剛才的文件中異步讀取我們寫入的內容:

using System;using System.IO;using System.Text;using System.Threading;namespace AsyncFileRead{  class Program  {    const int maxsize = 1024;    static byte[] readbytes = new byte[maxsize];    static void Main(string[] args)    {      ThreadPool.SetMaxThreads(1000, 1000);      PrintMessage("Main Thread start");      // 初始化FileStream對象      FileStream filestream = new FileStream("test.txt", FileMode.OpenOrCreate, FileAccess.ReadWrite, FileShare.ReadWrite, 100, false);      // 異步讀取文件內容      filestream.BeginRead(readbytes, 0, readbytes.Length, new AsyncCallback(EndReadCallback), filestream);      Console.Read();    }    private static void EndReadCallback(IAsyncResult asyncResult)    {      Thread.Sleep(1000);      PrintMessage("Asynchronous Method start");      // 把AsyncResult.AsyncState轉換為State對象      FileStream readstream = (FileStream)asyncResult.AsyncState;      int readlength = readstream.EndRead(asyncResult);      if (readlength <=0)      {        Console.WriteLine("Read error");        return;      }      string readmessage = Encoding.Unicode.GetString(readbytes, 0, readlength);      Console.WriteLine("Read Message is :" + readmessage);      readstream.Close();    }    // 打印線程池信息    private static void PrintMessage(String data)    {      int workthreadnumber;      int iothreadnumber;      // 獲得線程池中可用的線程,把獲得的可用工作者線程數量賦給workthreadnumber變量      // 獲得的可用I/O線程數量給iothreadnumber變量      ThreadPool.GetAvailableThreads(out workthreadnumber, out 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());    }  }}

運行結果:

C#,線程處理,線程池,I/O線程

這里有個需要注意的問題:如果大家測試的時候, 應該把開始生成的text.txt文件放到該工程下bin/debug/目錄下, 我剛開始的做的時候就忘記拷過去的, 讀出來的數據長度一直為0(這里我犯的錯誤寫下了,希望大家可以注意,也是警惕自己要小心。)

二、I/O線程實現對請求的異步

我們同樣可以利用I/O線程來模擬對瀏覽器對服務器請求的異步操作,在.net類庫中的WebRequest類提供了異步請求的支持,

下面就來演示下如何實現請求異步:

using System;using System.Net;using System.Threading;namespace RequestSample{  class Program  {    static void Main(string[] args)    {      ThreadPool.SetMaxThreads(1000, 1000);      PrintMessage("Main Thread start");      // 發出一個異步Web請求      WebRequest webrequest =WebRequest.Create("http://www.cnblogs.com/");      webrequest.BeginGetResponse(ProcessWebResponse, webrequest);      Console.Read();    }    // 回調方法    private static void ProcessWebResponse(IAsyncResult result)    {      Thread.Sleep(500);      PrintMessage("Asynchronous Method start");      WebRequest webrequest = (WebRequest)result.AsyncState;      using (WebResponse webresponse = webrequest.EndGetResponse(result))      {              Console.WriteLine("Content Length is : "+webresponse.ContentLength);      }    }    // 打印線程池信息    private static void PrintMessage(String data)    {      int workthreadnumber;      int iothreadnumber;      // 獲得線程池中可用的線程,把獲得的可用工作者線程數量賦給workthreadnumber變量      // 獲得的可用I/O線程數量給iothreadnumber變量      ThreadPool.GetAvailableThreads(out workthreadnumber, out 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());    }  }}

運行結果為:

 C#,線程處理,線程池,I/O線程

寫到這里這篇關于I/O線程的文章也差不多寫完了, 其實I/O線程還可以做很多事情,在網絡(Socket)編程,web開發中都會用I/O線程,本來想寫個Demo來展示多線程在實際的工作中都有那些應用的地方的, 但是后面覺得還是等多線程系列都講完后再把知識一起串聯起來做個Demo會好點,至于后面文章中將介紹下線程同步的問題。



注:相關教程知識閱讀請移步到c#教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 克山县| 舟曲县| 灵武市| 澄江县| 宁城县| 新干县| 布尔津县| 洛南县| 麟游县| 肇州县| 兴山县| 镇平县| 喀喇| 澳门| 即墨市| 长子县| 桐乡市| 博兴县| 分宜县| 陵川县| 延川县| 泸水县| 廊坊市| 长顺县| 诏安县| 休宁县| 浙江省| 呈贡县| 莒南县| 仪陇县| 吉木乃县| 南投市| 乐亭县| 岢岚县| 康马县| 沙坪坝区| 昌图县| 青川县| 永春县| 信宜市| 绥江县|