多線程編程學習筆記——線程池(二)

来源:http://www.cnblogs.com/chillsrc/archive/2017/11/13/7824936.html
-Advertisement-
Play Games

如果我們要從線程池中取消某個線程的操作,應該如何實現呢?本示例使用CancellationTokenSource和CancellationToken兩個類來實現在取消線程池中的操作。 ...


接上文 多線程編程學習筆記——線程池(一)

 

三、線程池與並行度

此示例是學習如何應用線程池實現大量的操作,及與創建大量線程進行工作的區別。

1. 代碼如下

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading; 

namespace ThreadPoolDemo
{  

    class Program
    {   

        static void Main(string[] args)
        {

            Console.WriteLine("開始測試線程池與自創線程。。。");
            const int total = 500;
            long millisecondes = 0;
            Stopwatch sw = new Stopwatch();
            sw.Start();
            ThreadRun(total);
            sw.Stop();
            millisecondes = sw.ElapsedMilliseconds;
            decimal mom = (decimal)(Environment.WorkingSet / (1024 * 1024.0));            sw.Reset();

            sw.Start();
            ThreadPoolRun(total);

            sw.Stop();
            Console.WriteLine("自創線程總耗時 {0},占用記憶體:{1} MB", millisecondes,mom);
            Console.WriteLine("線程池總耗時 {0} ,占用記憶體:{1} MB", sw.ElapsedMilliseconds, Environment.WorkingSet / (1024 * 1024.0));
            Console.Read();

        }

        private static void  ThreadRun(int total)
        {
            using (var countdown = new CountdownEvent(total))
            {
                Console.WriteLine("開始--自創線程。。。");
                for (int i = 0; i < total; i++)
                {
                    var t = new Thread(() =>
                    {
                        Console.WriteLine("自創線程ID :{0}", Thread.CurrentThread.ManagedThreadId);
                        Thread.Sleep(TimeSpan.FromSeconds(0.1));
                        countdown.Signal();//向 CountdownEvent 註冊信號,同時減小 CurrentCount 的值。

                    });
                    t.Start();
                }
                countdown.Wait();  // 阻塞當前線程,直到 CountdownEvent 的信號數量變為 0
                Console.WriteLine("-----------------");
            }
        }

        private static void ThreadPoolRun(int total)
        {
            using (var countdown = new CountdownEvent(total))

            {
                Console.WriteLine("開始--線程池。。。");
                for (int i = 0; i < total; i++)
                {
                    ThreadPool.QueueUserWorkItem(_ =>
                    {
                        Console.WriteLine("線程池工作線程ID :{0}", Thread.CurrentThread.ManagedThreadId);
                        Thread.Sleep(TimeSpan.FromSeconds(0.1));
                        countdown.Signal();//向 CountdownEvent 註冊信號,同時減小 CurrentCount 的值。
                    });                 

                }
                countdown.Wait();  // 阻塞當前線程,直到 CountdownEvent 的信號數量變為 0
                Console.WriteLine("-----------------");

            }
        }
    }
}

 

2.程式運行結果如下圖。

 

    1) 這個示例中我們自己創建了500個線程,每個線程一個操作,每個線程都阻塞100毫秒。總計耗時  11秒,消耗資源如下圖。

 

    2)我們使用線程池執行相同的500個操作。總計耗時  9秒,消耗資源如下圖。

 

從1)與2)的比較上可以看出來,自創線程消耗的CPU資源比線程池要多。

 

四、 從線程池中取消操作

如果我們要從線程池中取消某個線程的操作,應該如何實現呢?本示例使用CancellationTokenSource和CancellationToken兩個類來實現在取消線程池中的操作。這兩個是是在net 4.0引入的。

 1.示例代碼

using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Text;
using System.Threading; 

namespace ThreadPoolDemo
{

    class Program
    {    

        static void Main(string[] args)
        {
            Console.WriteLine("開始測試線程池中取消正在運行的線程。。。");
            using (var cts = new CancellationTokenSource())
            {
                CancellationToken token = cts.Token;
                ThreadPool.QueueUserWorkItem(_ => AsyncOper(token));
                Thread.Sleep(TimeSpan.FromSeconds(2));
                cts.Cancel();//取消線程操作
            }

            using (var cts = new CancellationTokenSource())
            {
                CancellationToken token = cts.Token;
                ThreadPool.QueueUserWorkItem(_ => AsyncOperation(token));
                Thread.Sleep(TimeSpan.FromSeconds(2));
                cts.Cancel();//取消線程操作

            }
            using (var cts = new CancellationTokenSource())

            {
                CancellationToken token = cts.Token;
                ThreadPool.QueueUserWorkItem(_ => AsyncOper3(token));

                Thread.Sleep(TimeSpan.FromSeconds(2));
                cts.Cancel();//取消線程操作
            }

            Thread.Sleep(TimeSpan.FromSeconds(2));
            Console.WriteLine("。。。。。。。。。。取消正在運行的線程結束。。。。。。");
            Console.Read();
        }

        private static void  AsyncOperation(CancellationToken token)
        {

            try
            {
                Console.WriteLine("開始--線程池中的第二個工作線程。。。");
                for (int i = 0; i < 5; i++)

                {
                    token.ThrowIfCancellationRequested();//獲取取消請求,拋出OperationCanceledException異常,
                    Thread.Sleep(TimeSpan.FromSeconds(1));
                }

                Console.WriteLine("-------線程池中的第二個工作線程 工作完成----------");
            }
            catch (OperationCanceledException ex)
            {
                Console.WriteLine("使用拋出異常方法取消第二個工作線程  ID:{0},{1}", Thread.CurrentThread.ManagedThreadId,ex.Message);
            }
        }
        private static void AsyncOper(CancellationToken token)
        {

            Console.WriteLine("開始--線程池中的第一個工作線程。。。");
            for (int i = 0; i < 5; i++)
            {
                if (token.IsCancellationRequested)//判斷是否已經取消操作
                {
                    Console.WriteLine("使用輪詢方法取消工作線程  ID:{0}", Thread.CurrentThread.ManagedThreadId);
                    return;
                }
                Thread.Sleep(TimeSpan.FromSeconds(1));
            }
            Console.WriteLine("-------線程池中的第一個工作線程 工作完成----------");
        }
        private static void AsyncOper3(CancellationToken token)
        {
            Console.WriteLine("開始--線程池中的第三個工作線程。。。");
            bool cancel = false;
            token.Register(()=>cancel = true);
            for (int i = 0; i < 5; i++)
            {
                if (cancel)//判斷是否已經取消操作
                {
                    Console.WriteLine("通過註冊回調函數取消第三個工作線程  ID:{0}", Thread.CurrentThread.ManagedThreadId);
                    return;
                }
                Thread.Sleep(TimeSpan.FromSeconds(1));
            }   

            Console.WriteLine("-------線程池中的第三個工作線程 工作完成----------");
        }
    }
}

2.運行結果如下圖。

 

本示例一共實現了三種取消線程池中操作的方式。

  1. 輪詢檢查CancellationToken.IsCancellationRequested屬性,如果為true,則說明操作被取消。
  2. 拋出一個OperationCancellationException異常。這允許操作之外的代碼來取消操作。
  3. 註冊一個回調函數,當操作取消時,線程池將調用回調函數,這樣做的好處是將取消操作邏輯傳遞到另一個非同步操作中。

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • lamp環境已搭建完成基礎之上 上傳owncloud-10.0.3.zip到Linux 解壓owncloud-10.0.3.zip [root@localhost ~]#unzip owncloud-10.0.3.zip 複製解壓後的文件到/var/www/html [root@localhost ...
  • 要想從現在的低薪(年薪10萬以下)快讀變成未來的高新(年薪30萬以上)我們要做的就只有從自身改變開始! 人改變自己的勇氣,朱啊喲取決於我們自己當前的痛苦程度!直到某一天真的迴避不了了,才會被動的改變,被動改變結果往往是來不及的,或者花的代價相當高,獲得的回報又比較低!改變自己越早越好!成功的人一般都 ...
  • 本文需要實現的功能如下:某文件夾下具有由按數字編號命名的文件夾,需要刪除除最大編碼外的文件。 具體實現 大致思路:迴圈遍歷該文件夾下所有文件,正則匹配出最大編碼文件;然後迴圈文件,刪除除最大編碼外的文件。 實現代碼如下: 實現效果:文件夾:/root/cloud/builds 執行腳本後: 用到的S ...
  • Mysql安裝 CentOS 7 版本將MySQL資料庫軟體從預設的程式列表中移除,用mariadb代替了。MariaDB資料庫管理系統是MySQL的一個分支,主要由開源社區在維護,採用GPL授權許可。開發這個分支的原因之一是:甲骨文公司收購了MySQL後,有將MySQL閉源的潛在風險,因此社區採用 ...
  • Memcached 是一個高性能的分散式記憶體對象緩存系統,用於動態Web應用以減輕資料庫負載。它通過在記憶體中緩存數據和對象來減少讀取資料庫的次數,從而提高動態、資料庫驅動網站的速度。Memcached基於一個存儲鍵/值對的hashmap。其守護進程(daemon )是用寫的,但是客戶端可以用任何語言 ...
  • ABP是“ASP.NET Boilerplate Project (ASP.NET樣板項目)”的簡稱。ASP.NET Boilerplate是一個用最佳實踐和流行技術開發現代WEB應用程式的新起點,它旨在成為一個通用的WEB應用程式框架和項目模板。框架ABP是基於最新的ASP.NET CORE,AS ...
  • 一、系統介紹 本軟體主要讓你的文件夾多彩多樣化,而不是千篇一律黃色的文件夾,相信豐富多彩的顏色更便於記憶、能讓心情愉快些。 文件夾顏色眾多,到底有多少種呢,10種? 100種? 1000種? NO,精確計算是255*255*255 , 只是很多顏色很相近, 而能夠明顯有差別的不是很多,為了便於自己記 ...
  • 在ABP框架下使用NHibernate和Dapper實現資料庫訪問。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...