色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

多線程之線程池

汽車電子技術(shù) ? 來源:程序猿知秋 ? 作者:程序猿知秋 ? 2023-02-28 09:53 ? 次閱讀

圖片

線程池的基本概念

什么是線程池?

  • ** .NET Framework的ThreadPool類提供一個(gè)線程池**
  • “線程池”是可以用來在后臺(tái)執(zhí)行多個(gè)任務(wù)的線程集合
  • 線程池通常用于服務(wù)器應(yīng)用程序。 每個(gè)傳入請(qǐng)求都將分配給線程池中的一個(gè)線程,因此可以異步處理請(qǐng)求,而不會(huì)占用主線程,也不會(huì)延遲后續(xù)請(qǐng)求的處理
  • 一旦池中的某個(gè)線程完成任務(wù),它將返回到等待線程隊(duì)列中,等待被再次使用。 這種重用使應(yīng)用程序可以避免為每個(gè)任務(wù)創(chuàng)建新線程的開銷
  • 線程池通常具有最大線程數(shù)限制。 如果所有線程都繁忙,則額外的任務(wù)將放入隊(duì)列中,直到有線程可用時(shí)才能夠得到處理

為什么要用線程池?

  • 線程是非常消耗資源的,如果我們每次需要子線程來執(zhí)行任務(wù),就去創(chuàng)建一個(gè)新的線程,那當(dāng)我們執(zhí)行1千次1萬次甚至是100萬次的時(shí)候,那么電腦的資源消耗就非常嚴(yán)重,甚至承受不了
  • 線程池可以避免大量的創(chuàng)建和銷毀的開支,具有更好的性能和穩(wěn)定性
  • 開發(fā)人員把線程交給系統(tǒng)管理,可以集中精力處理其他任務(wù)

前臺(tái)線程&后臺(tái)線程

  • 前臺(tái)線程: 當(dāng)程序運(yùn)行起來后,主線程已經(jīng)運(yùn)行結(jié)束了,但是程序卻沒有停止,子線程依然在運(yùn)行
  • 后臺(tái)線程: 只要主線程和所有的前臺(tái)線程執(zhí)行結(jié)束,就算后臺(tái)線程的任務(wù)還沒完成,也會(huì)強(qiáng)行打斷直接退出
  • 特點(diǎn): 后臺(tái)適用于不太重要的任務(wù),即被中斷了也沒事的。 前臺(tái)線程適用于比較重要的任務(wù),必須等任務(wù)執(zhí)行完程序才能關(guān)閉
  • ThreadPool 線程池中的線程都是 后臺(tái)線程

線程池的使用

設(shè)置線程池大小

  • ThreadPool.SetMaxThreads (int workerThreads,int completionPortThreads)
  • ThreadPool.SetMinThreads (int workerThreads,int completionPortThreads)
  • 參數(shù)解析:
    • completionPortThreads:線程池中異步 I/O 線程的數(shù)目 (I/O 完成線程)
    • workerThreads:線程池中輔助線程的數(shù)目**(工作線程)**
  • 對(duì)于以上兩個(gè)參數(shù)的解釋,摘自網(wǎng)絡(luò)其它博客:
    • 主要用來完成 輸入和輸出的工作 ,在這種情況下, 計(jì)算機(jī)需要I/O設(shè)備完成輸入和輸出的任務(wù)。在處理過程中, CPU是不需要參與處理過程的此時(shí)正在運(yùn)行的線程將處于等待狀態(tài)只有等任務(wù)完成后才會(huì)有事可做這樣就造成線程資源浪費(fèi)的問題。為了解決這樣的問題,可以通過線程池來解決這樣的問題,讓線程池來管理線程

    • .NET中的一些API方法,通過APM(異步編程模式),內(nèi)部實(shí)現(xiàn)了ThreadPool.BindHandle方法。BeginXXX方法將用戶的回調(diào)委托送到某個(gè)設(shè)備驅(qū)動(dòng)程序,然后返回線程池。

      當(dāng)做完成后,OS會(huì)通過IOCP提醒CLR它工作已經(jīng)完成,當(dāng)接收到通知后,I/O線程會(huì)醒來并且運(yùn)行用戶的回調(diào)

    • 所以工作線程由開發(fā)人員調(diào)用, I/O線程由CLR調(diào)用 。所以通常情況下,開發(fā)者并不會(huì)直接用到它。因此可以認(rèn)為, 工作者線程和I/O線程沒有區(qū)別,它們都是普通的線程但是CLR線程池中區(qū)分它們的目的是為了避免線程都去處理I/O回調(diào)而被耗盡,從而引發(fā)死鎖 。(設(shè)想,所有的工作者線程每一個(gè)都去等待I/O異步完成。)

    • 用來完成一些 計(jì)算的任務(wù) ,在任務(wù)執(zhí)行的過程中,需要 CPU不間斷地處理 ,所以,在工作者線程的執(zhí)行過程中,CPU和線程的資源是充分利用的

    • .NET中的術(shù)語工作者線程指的是任何線程而不是僅僅主線程

    • 工作者線程

    • I/O線程

啟動(dòng)線程任務(wù)使用: QueueUserWorkItem()方法

  • static bool QueueUserWorkItem(WaitCallback callBack):參數(shù)為一個(gè)帶一個(gè)object類型參數(shù)的委托,最后返回bool值成功則返回true
class ThreadPoolTest
{
  static void Main()
  {
    Console.WriteLine("啟動(dòng)多線程...");
    for(int i = 0; i <10; i++)
    {
      ThreadPool.QueueUserWorkItem( p => printStr("當(dāng)前線程") );
    }
    Console.WriteLine("結(jié)束多線程...");
    Console.ReadKey();
  }


  private static void printStr(string str)
  {
    Console.WriteLine("{0}是:{1}", str, Thread.CurrentThread.ManagedThreadId);
  }
}

輸出結(jié)果, 可以看到有很多線程ID是重復(fù)的,這就是線程池的強(qiáng)大之處了

啟動(dòng)多線程...
結(jié)束多線程...
當(dāng)前線程是:8
當(dāng)前線程是:6
當(dāng)前線程是:10
當(dāng)前線程是:9
當(dāng)前線程是:7
當(dāng)前線程是:11
當(dāng)前線程是:12
當(dāng)前線程是:9
當(dāng)前線程是:6
當(dāng)前線程是:10

查看最大/最小線程數(shù) && 設(shè)置最大/最小線程數(shù)

class ThreadPoolTest
{
  static void Main()
  {
    // 獲取默認(rèn)的線程池中的最大,最小線程數(shù)
    ThreadPool.GetMaxThreads(out int maxWorkerThreads, out int maxCompletionPortThreads);
    Console.WriteLine("最大線程數(shù),工作線程:{0}, IO線程數(shù):{1}", maxWorkerThreads, maxCompletionPortThreads);
    ThreadPool.GetMinThreads(out int minWorkerThreads, out int minCompletionPortThreads);
    Console.WriteLine("最小線程數(shù),工作線程:{0}, IO線程數(shù):{1}", minWorkerThreads, minCompletionPortThreads);


    // 重新設(shè)置最大、最小線程數(shù)
    ThreadPool.SetMaxThreads(10, 10);
    ThreadPool.SetMinThreads(3, 3);
    // 獲取默認(rèn)的線程池中的最大,最小線程數(shù)
    ThreadPool.GetMaxThreads(out int newMaxWorkThread, out int newMaxIOThread);
    Console.WriteLine("重新設(shè)置后的最大線程數(shù),工作線程:{0}, IO線程數(shù):{1}", newMaxWorkThread, newMaxIOThread);
    ThreadPool.GetMinThreads(out int newMinWorkThread, out int newMinIOThread);
    Console.WriteLine("重新設(shè)置后的最小線程數(shù),工作線程:{0}, IO線程數(shù):{1}", newMinWorkThread, newMinIOThread);




    Console.WriteLine("啟動(dòng)多線程...");
    for(int i = 0; i <10; i++)
    {
      ThreadPool.QueueUserWorkItem( p => printStr("當(dāng)前線程") );
    }
    Console.WriteLine("結(jié)束多線程...");
    Console.ReadKey();
  }


  private static void printStr(string str)
  {
    Console.WriteLine("{0}是:{1}", str, Thread.CurrentThread.ManagedThreadId);
  }
}

輸出結(jié)果

最大線程數(shù),工作線程:32767, IO線程數(shù):1000
最小線程數(shù),工作線程:8, IO線程數(shù):8
重新設(shè)置后的最大線程數(shù),工作線程:10, IO線程數(shù):10
重新設(shè)置后的最小線程數(shù),工作線程:3, IO線程數(shù):3
啟動(dòng)多線程...
結(jié)束多線程...
當(dāng)前線程是:6
當(dāng)前線程是:9
當(dāng)前線程是:7
當(dāng)前線程是:8
當(dāng)前線程是:11
當(dāng)前線程是:10
當(dāng)前線程是:12
當(dāng)前線程是:11
當(dāng)前線程是:7
當(dāng)前線程是:9

ThreadPool.SetMaxThreads的默認(rèn)值

  • 它取決于.NET框架版本,在2.0,3.0和4.0中進(jìn)行了更改。 在2.0中它是核心數(shù)量的50倍。 在3.0(又名2.0 SP1)中,它是內(nèi)核數(shù)量的250倍,4.0根據(jù)位數(shù)和操作系統(tǒng)資源使其動(dòng)態(tài)化。 默認(rèn) Max I / O完成線程是1000
  • 一般來說,它是非常的高,一個(gè)程序永遠(yuǎn)不會(huì)接近

使用以上方法設(shè)置線程數(shù)據(jù)大小時(shí)需注意

  • 不能將最大工作線程數(shù)或 I/O 完成線程數(shù)設(shè)置為小于計(jì)算機(jī)上的處理器數(shù)的數(shù)字
  • 不能將最大工作線程數(shù)或 I/O 完成線程數(shù)設(shè)置為小于相應(yīng)最小工作線程數(shù)或 I/O 完成線程數(shù)的數(shù)字
  • 默認(rèn)情況下,最小線程數(shù)設(shè)置為系統(tǒng)上的處理器數(shù)。 可以使用該方法 SetMinThreads 增加最小線程數(shù)。 但是,不必要地增加這些值可能導(dǎo)致性能問題。 如果在同一時(shí)間開始太多的任務(wù),則所有任務(wù)均可能會(huì)很慢。 在大多數(shù)情況下,線程池將使用自己的算法更好地分配線程。 將最小處理器減少到小于處理器數(shù)也會(huì)損害性能

查看系統(tǒng)cpu的處理數(shù)

圖片編輯

**線程等待 **

由于線程池中的線程都是后臺(tái)線程,當(dāng)主線程及所有前臺(tái)線程執(zhí)行完時(shí),后臺(tái)線程就會(huì)被中斷,但如果我們希望 主線程等待 線程池執(zhí)行完成任務(wù)后再關(guān)閉程序 怎么辦呢?

需要使用到ManualResetEvent類

圖片編輯

**ManualResetEvent需要一個(gè)bool類型的參數(shù)來表示暫停和停止

**

class ThreadPoolTest2
{
  // 參數(shù)設(shè)置為false
  static ManualResetEvent manualResetEvent = new ManualResetEvent(false);

  static void Main()
  {
    Console.WriteLine("啟動(dòng)多線程...");
    ThreadPool.QueueUserWorkItem(p => printStr("當(dāng)前線程"));
    // 等待 線程池執(zhí)行任務(wù)完成
    manualResetEvent.WaitOne();
    Console.WriteLine("結(jié)束多線程...");
  }


  private static void printStr(string str)
  {
    Console.WriteLine("{0}是:{1}", str, Thread.CurrentThread.ManagedThreadId);
    // 設(shè)置為true
    manualResetEvent.Set();
  }
}

輸出結(jié)果

啟動(dòng)多線程...
當(dāng)前線程是:6
結(jié)束多線程...

ManualResetEvent類的參數(shù)值執(zhí)行順序如下:

  • false-- WaitOne等待
  • true--WaitOne通過

注:一般情況下,不要阻塞線程池中的線程,因?yàn)?/strong>寫代碼無意間造成的線程等待沒有釋放,一旦線程池線程耗盡就會(huì)形成死鎖

造成死鎖的案例: 設(shè)置了最大線程數(shù)是9,循環(huán)創(chuàng)建15個(gè)線程,意味著 后6個(gè)線程必然需要利用線程池中前面用過的線程, 但是由于前面創(chuàng)建的線程都直接讓阻塞了,沒有釋放,這時(shí)循環(huán)到第10個(gè)線程時(shí)由于沒有空閑線程,線程池沒法執(zhí)行就直接跳過了,主線程會(huì)繼續(xù)執(zhí)行后面的循環(huán),這樣也永遠(yuǎn)不會(huì) 執(zhí)行 manualResetEvent.Set() 方法,最后就成死鎖了

private static void Test1()
{
  // 設(shè)置最大線程
  ThreadPool.SetMaxThreads(9, 9);
  // 設(shè)置最小線程
  ThreadPool.SetMinThreads(3, 3);
  ManualResetEvent manualResetEvent = new ManualResetEvent(false);
  for (int i = 0; i < 15; i++)
  {
    int k = i;
    ThreadPool.QueueUserWorkItem(p =>
    {
      Console.WriteLine(k);
      if (k < 10)
      {
        manualResetEvent.WaitOne();
      }
      else
      {
        // 設(shè)為true
        manualResetEvent.Set();
      }
    });
  }
  if (manualResetEvent.WaitOne())
  {
    Console.WriteLine("沒有死鎖。。。。。。。。。");
  }
  Console.WriteLine("結(jié)束。。。。。。。。。。");
}

輸出結(jié)果

1
0
2
3
4
5
6
7
8
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 服務(wù)器
    +關(guān)注

    關(guān)注

    12

    文章

    9123

    瀏覽量

    85324
  • 應(yīng)用程序
    +關(guān)注

    關(guān)注

    37

    文章

    3265

    瀏覽量

    57677
  • 線程池
    +關(guān)注

    關(guān)注

    0

    文章

    57

    瀏覽量

    6844
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    C語言線程的實(shí)現(xiàn)方案

    這是一個(gè)簡(jiǎn)單小巧的C語言線程實(shí)現(xiàn),在 Github 上有 1.1K 的 star,很適合用來學(xué)習(xí) Linux 的多線程編程。
    的頭像 發(fā)表于 01-29 16:43 ?1535次閱讀

    跨平臺(tái)的線程組件--TP組件

    /銷毀代價(jià)是很高的。那么我們要怎么去設(shè)計(jì)多線程編程呢???答案:對(duì)于長駐的線程,我們可以創(chuàng)建獨(dú)立的線程去執(zhí)行。但是非長駐的線程,我們可以通過線程
    的頭像 發(fā)表于 04-06 15:39 ?871次閱讀

    多線程程之一: 問題提出

    多線程程之一 問題提出編寫一個(gè)耗時(shí)的單線程程序:  新建一個(gè)基于對(duì)話框的應(yīng)用程序SingleThread,在主對(duì)話框IDD_SINGLETHREAD_DIALOG添加一個(gè)按鈕,ID為
    發(fā)表于 10-22 11:41

    多線程程之線程間通訊

    多線程程之線程間通訊七、線程間通訊  一般而言,應(yīng)用程序中的一個(gè)次要線程總是為主線程執(zhí)行特
    發(fā)表于 10-22 11:43

    多線程程之線程的同步

    多線程程之線程的同步八、線程的同步  雖然多線程能給我們帶來好處,但是也有不少問題需要解決。例如,對(duì)于像磁盤驅(qū)動(dòng)器這樣獨(dú)占性系統(tǒng)資源,
    發(fā)表于 10-22 11:43

    Delphi教程之多線程與數(shù)據(jù)庫

    Delphi教程之多線程與數(shù)據(jù)庫,很好的Delphi資料,快來下載學(xué)習(xí)吧。
    發(fā)表于 04-11 15:59 ?5次下載

    多線程好還是單線程好?單線程多線程的區(qū)別 優(yōu)缺點(diǎn)分析

    摘要:如今單線程多線程已經(jīng)得到普遍運(yùn)用,那么到底多線程好還是單線程好呢?單線程多線程的區(qū)別又
    發(fā)表于 12-08 09:33 ?8.1w次閱讀

    mfc多線程編程實(shí)例及代碼,mfc多線程間通信介紹

    摘要:本文主要以MFC多線程為中心,分別對(duì)MFC多線程的實(shí)例、MFC多線程之間的通信展開的一系列研究,下面我們來看看原文。
    發(fā)表于 12-08 15:23 ?1.8w次閱讀
    mfc<b class='flag-5'>多線程</b>編程實(shí)例及代碼,mfc<b class='flag-5'>多線程</b>間通信介紹

    從CPU說起多線程以及線程

    從這篇開始將會(huì)開啟高性能、高并發(fā)系列,本篇是該系列的開篇,主要關(guān)注多線程以及線程。 一切要從CPU說起 你可能會(huì)有疑問,講多線程為什么要從CPU說起呢?原因很簡(jiǎn)單,在這里沒有那些時(shí)髦
    的頭像 發(fā)表于 03-02 13:48 ?1926次閱讀

    PyQT5+OpenCV多線程協(xié)作演示

    學(xué)習(xí)多線程最典型的問題就是如何在多個(gè)線程之間傳遞消息與寫作,PyQT5的線程支持在不同線程之間傳遞信號(hào)觸發(fā)事件,實(shí)現(xiàn)多個(gè)線程之間的協(xié)助,完成
    的頭像 發(fā)表于 03-08 14:58 ?1334次閱讀

    如何用C++實(shí)現(xiàn)一個(gè)線程呢?

    C++線程是一種多線程管理模型,把線程分成任務(wù)執(zhí)行和線程調(diào)度兩部分。
    發(fā)表于 06-08 14:53 ?1765次閱讀
    如何用C++實(shí)現(xiàn)一個(gè)<b class='flag-5'>線程</b><b class='flag-5'>池</b>呢?

    Spring 的線程應(yīng)用

    我們?cè)谌粘i_發(fā)中,經(jīng)常跟多線程打交道,Spring 為我們提供了一個(gè)線程方便我們開發(fā),它就是 ThreadPoolTaskExecutor ,接下來我們就來聊聊 Spring 的線程
    的頭像 發(fā)表于 10-13 10:47 ?620次閱讀
    Spring 的<b class='flag-5'>線程</b><b class='flag-5'>池</b>應(yīng)用

    線程基本概念與原理

    、17、20等的新特性,簡(jiǎn)化了多線程編程的實(shí)現(xiàn)。 提高性能與資源利用率 線程主要解決兩個(gè)問題:線程創(chuàng)建與銷毀的開銷以及線程競(jìng)爭(zhēng)造成的性能瓶
    的頭像 發(fā)表于 11-10 10:24 ?528次閱讀

    什么是動(dòng)態(tài)線程?動(dòng)態(tài)線程的簡(jiǎn)單實(shí)現(xiàn)思路

    因此,動(dòng)態(tài)可監(jiān)控線程一種針對(duì)以上痛點(diǎn)開發(fā)的線程管理工具。主要可實(shí)現(xiàn)功能有:提供對(duì) Spring 應(yīng)用內(nèi)線程
    的頭像 發(fā)表于 02-28 10:42 ?639次閱讀

    java實(shí)現(xiàn)多線程的幾種方式

    了多種實(shí)現(xiàn)多線程的方式,本文將詳細(xì)介紹以下幾種方式: 1.繼承Thread類 2.實(shí)現(xiàn)Runnable接口 3.Callable和Future 4.線程 5.Java 8中
    的頭像 發(fā)表于 03-14 16:55 ?686次閱讀
    主站蜘蛛池模板: 日韩a视频在线观看| 欧洲馒头大肥p| 欧美日韩一二区旡码高清在线| 亚洲精品天堂自在久久77| 搞av.com| 欧美精品一卡二卡| 99精品国产在热| 蜜臀色欲AV无人A片一区| 伊人久久中文| 亚洲地址一地址二地址三| 国产高清免费视频免费观看| 国产日韩欧美另类| 午夜理论片YY4399影院| 国产偷窥盗摄一区二区| 性西欧俄罗斯极品| 韩国无遮羞禁动漫在线观看| 亚洲AV无码乱码国产麻豆P| 国精产品一区二区三区有限公司 | 国产午夜精品不卡视频| 亚洲AV色香蕉一区二区三区| 国产在线精品视频资源| 亚洲一区二区三区高清网| 久久性色AV亚洲电影无码| 97伦理97伦理2018最新| 青青青青草| 国产福利秒拍weipai.ee| 亚洲另类中文字幕| 久久久久久久电影| 99久久久无码国产精品免费人妻| 青娱乐极品视觉盛宴国产视频| 国产成人免费视频| 亚洲精品一二三区区别在哪| 久久婷婷丁香五月色综合啪免费| 8x8x我要打机飞在线观看| 日日碰狠狠添天天爽| 国产专区亚洲欧美另类在线| 2023国产精品一卡2卡三卡4卡| 日韩精品久久久久久久电影| 国产伊人自拍| 99精品视频| 亚州AV中文无码乱人伦在线|