C# 多線程系列(一)

来源:https://www.cnblogs.com/wrbxdj/archive/2018/02/23/8459696.html
-Advertisement-
Play Games

線程是怎樣工作的 1.多線程由一個線程調度器來進行內部管理,一個功能是CLR常常委托給操做系統。 一個線程調度器確保所有激活的線程在執行期間被合適的分配,等待或者阻塞的線程(比如,一個獨占鎖或者等待用戶輸入)不占用CPU資源。 2.在單核電腦上,一個線程調度器讓時間片在每一個激活的線程中切換。在wi ...


線程是怎樣工作的

1.多線程由一個線程調度器來進行內部管理,一個功能是CLR常常委托給操做系統。

一個線程調度器確保所有激活的線程在執行期間被合適的分配,等待或者阻塞的線程(比如,一個獨占鎖或者等待用戶輸入)不占用CPU資源。

2.在單核電腦上,一個線程調度器讓時間片在每一個激活的線程中切換。在windows操作系統下,線程切換的時間分片通常為10微秒,遠遠大於CPU的開銷時間(通常小於1微秒)。

3.在一個多核的電腦上,多線程實現了一個混合的時間片和真正的併發,不同的線程同時在不同的CPU上執行代碼。還是存在某些時間片,因為操作系統需要服務它自己的線程,包括其他的應用的線程。

4.當一個線程的執行被內部因素打斷,比如時間片,則說這個線程是搶占式的。在大部分情形下,一個線程不能控制自己何時何地被搶占。

進程與線程

線程並行運行在一個單獨的進程中。進程之間是完全隔離的;線程在一定程度上隔離。運行在同一個應用程式下的線程共用堆記憶體。

進程間的切換:

先載入程式A的上下文,然後開始執行A,保存程式A的上下文,調入下一個要執行的程式B的程式上下文,然後開始執行B,保存程式B的上下文。。。

線程間的切換:

先載入程式A的上下文,然後執行A的A1線程,再執行A的A2線程。。。保存程式A的上下文;

1.線程結束無法重啟

class Program
    {
        static void Main(string[] args)
        {
            Thread thread = new Thread(threadOne);
            thread.Start();
            Thread.Sleep(5000);
            //thread.Start();打開報錯
        }

        static void threadOne()
        {
            for(int i=0; i<100; i++)
            {
                lib.put("t"+i);
                Thread.Sleep(10);
            }
        }
    }

一旦開始,一個線程的IsAlive屬性返回true,直到這個線程結束。當傳遞給線程的構造函數的委托完成執行時,這個線程結束。一旦結束,這個線程不能重啟。

2.記憶體隔離

class Program
    {
        static void Main(string[] args)
        {
            Thread thread = new Thread(threadOne);
            thread.Start();
            threadOne();//主線程調用
            Thread.Sleep(5000);
            //thread.Start();打開報錯
        }

        static void threadOne()
        {
            for(int i=0; i<10; i++)
            {
                lib.put("t");
                Thread.Sleep(10);
            }
        }
    }

  CLR給每個線程分配自己記憶體棧,因此局部變數可以保持分離。所以兩次調用threadOne都正常列印了10個"t"。

3. 數據共用

    class Program
    {
        static bool bDone = false;
        static void Main(string[] args)
        {
            Thread thread = new Thread(threadOne);
            thread.Start();
            threadOne();
            Console.ReadKey();
        }

        static void threadOne()
        {
            if(!bDone)
            {
                bDone = true;
                lib.put("threadOne");
            }
        }
    }

 

如果多個線程對同一個對象實例有相同的引用,這些線程就共用這個對象實例的數據。所以只會輸出一個threadOne。

4.線程安全

上述例子,是有可能列印出兩次threadOne的。當bDone被置成true之前,若主線程和子線程都已經通過if判定,將列印兩次。

 

static void threadOne()
{
    if(!bDone)
    {
        Thread.Sleep(50);//加上sleep後,會出現主線程和子線程都通過if判定後才把bDone置成true,所以列印兩次threadOne,
     bDone = true;
     lib.put(
"threadOne");
  }
}

 

改進方式當讀\寫一個公共欄位時,獲取一個獨占鎖(exclusive lock)。C#提供了關鍵字lock

class Program
    {
        static bool bDone = false;
        static readonly object locker = new object();
        static void Main(string[] args)
        {
            Thread thread = new Thread(threadOne);
            thread.Start();
            threadOne();
            Console.ReadKey();
        }

        static void threadOne()
        {
            lock (locker)
            {
                if (!bDone)
                {
                    Thread.Sleep(50);
                    bDone = true;
                    lib.put("threadOne");
                }
            }
        }
    }

 

當兩個線程同時搶占一個鎖時(在這個例子中,locker,一個線程等待,或者阻塞,直到這個鎖釋放。在這個例子中,這個鎖保證一次只有一個線程可以進入代碼的臨界區域,然後“threadOne”只會被列印一次。代碼在這種不確定的多線程背景下中被保護被叫做線程安全。

lock關鍵字將語句塊標記為關鍵部分,方法是獲取給定對象的互斥鎖,執行語句,然後釋放該鎖。在塊的開始處調用 Enter,而在塊的結尾處調用 Exit。這樣可確保當一個線程位於代碼的關鍵部分時,另一個線程不會進入該關鍵部分。 如果其他線程嘗試進入鎖定的代碼,則它將一直等待(即被阻止),直到該對象被釋放。一個線程,當阻塞的時候,不占用CPU資源。

 5.Join和Sleep

Join是一種同步方法,阻止調用線程 (即,調用的方法的線程),直到線程其Join方法調用已完成。 使用此方法以確保線程已終止。 如果線程不會終止,調用方將無限期阻止。

Join有多個重載方法,可以在Join方法中添加一個參數,milliseconds或者timeSpan。如果這個線程結束了則Join方法返回true,如果這個線程超時則返回false,但不結束該線程。

static void Main(string[] args)
{
    Thread thread = new Thread(threadOne);
    thread.Start();
    thread.Join();//等到thread線程結束後繼續本線程
    lib.print("end");
    Console.ReadKey();
}

static void threadOne()
{
    lib.print("threadOne");
    Thread.Sleep(3000);
}

 

Sleep暫停當前線程一段指定的時間。

Thread.Sleep(TimeSpan.FromHours(1));//sleep一個小時

Thread.Sleep(500);//500 線程被阻塞的毫秒數。 指定零0以指示應掛起此線程以使其他等待線程能夠執行。

當使用SleepJoin暫停線程時,這個線程是阻塞的,不消耗CPU資源。

 --------------原文連接--------------


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

-Advertisement-
Play Games
更多相關文章
  • 引言 隨著手機及移動設備的普及,移動端的應用也進入了熱潮。以前PC端的門戶網站,大多也均推出了適配移動設備的網站或者APP,再差的也註冊了個公眾號。在移動應用開發中,目前據我所瞭解到的解決方案有:1、原生態APP開發;2、適配移動設備的JS+HTML網站,套上跨平臺的“馬甲”;3、微信小程式及公眾號 ...
  • var array=['REG','2018','2018']; array.indexOf(‘REG’) // 0 array.indexOf(‘R’) // -1 array.indexOf(’2018′) // 1 array.indexOf(2018) // -1 var array=['R ...
  • Routing 到目前為止,我們已經解決了MVC的很多問題,但忽略了最基本的最重要的一個問題:當用戶發送請求時,會發生什麼? 最好的答案是“執行Action 方法”,但仍存在疑問:對於一個特定的URL請求,如何確定控制器和action 方法。在開始實驗31之前,我們首先來解答上述問題,你可能會困惑為 ...
  • 在前面的文章中,我們已經知道如何合併、拆分多個PDF文件,在這篇文章中的合併、拆分PDF文檔主要是以方便文檔管理的目的來操作文檔,在文檔查閱、管理及存儲上很方便實用。但是我們如果想要合併多個文檔中的部分文檔頁的內容,該如何來做呢?可以參考接下來將要介紹的合併方法。 PS: 本篇文章是對Free Sp ...
  • using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.... ...
  • IL DASM反編譯工具 使用C#的猿人或多或少都會對微軟的IL反編譯工具(ildasm.exe)有所認識。我最早接觸到這工具是公司同事使用他反編譯exe程式,進行研讀和修改。感覺他還是很強大。 IL是微軟平臺上的一門中間語言,我們常寫的C#代碼在編譯器中都會自動轉換成IL,然後在由即時編譯器(JI ...
  • 依照老周的良好作風,開始之前先說點題外話。 前面的博文中,老周介紹過自定義 MVC 視圖的搜索路徑,即向 ViewLocationFormats 列表添加相應的內容,其實,對 Razor Page 模型,也可以向 PageViewLocationFormats 列表添加相應的搜索路徑,比如 /MyP ...
  • 上一篇Redis的系列已經講了Redis的下載、安裝,接下來這一篇,主要講使用Redis提供的 ServiceStack.Redis 這個開發庫在C#項目中作為緩存服務使用的一個簡單示例,廢話不多話,直接上代碼。 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...