並行模式庫PPL應用實戰(一):使用task類創建並行任務

来源:http://www.cnblogs.com/xrunning/archive/2017/06/17/7039490.html
-Advertisement-
Play Games

自 VS2010 起,微軟就在 CRT 中集成了併發運行時(Concurrency Runtime),並行模式庫(PPL,Parallel Patterns Library)是其中的一個重要組成部分。7 年過去了,似乎大家都不怎麼Care這個事情,相關文章少少且多是蜻蜓點水。實際上這個庫的設計相當精 ...


自 VS2010 起,微軟就在 CRT 中集成了併發運行時(Concurrency Runtime),並行模式庫(PPL,Parallel Patterns Library)是其中的一個重要組成部分。7 年過去了,似乎大家都不怎麼Care這個事情,相關文章少少且多是蜻蜓點水。實際上這個庫的設計相當精彩,勝過 C++ 標準庫中 future/promise/async 系列許多,所以計劃寫一個系列探討 PPL 在實際項目中應用中的各種細節。

 

好了,從最簡單的代碼開始,先演示下如何使用 task 類和 lambda 表達式創建一個並行任務:

// final_answer.cpp
// compile with: /EHsc 

#include <ppltasks.h>
#include <iostream>

using namespace concurrency;
using namespace std;

int main(int argc, char *argv[])
{
    task<int> final_answer([]
    {
        return 42;
    });
    
    cout << "The final answer is: " << final_answer.get() << endl;
    
    return 0;
}

使用 Visual Studio 命令行工具編譯

cl /EHsc final_answer.cpp

執行結果為:

The final answer is: 42

 

task 類的原型如下:

template<typename _ReturnType>
class task;

其模板參數 _ReturnType 是任務返回值類型。 task:get 方法則用於獲取返回值,原型如下:

_ReturnType get() const;

 

task 類的構造函數原型:

template<typename T>
__declspec(noinline) explicit task(T _Param);

可以看到這是個模板函數,其參數 _Param 可以是 lambda 表達式、函數對象、仿函數、函數指針等可以以 _Param() 形式調用的類型,或者 PPL 中的 task_completion_event<result_type> 類型。因此可以使用各種靈活的方式構造 task 對象,其中 lambda 表達式無疑是最方便常用的一種。

 

接下來我們修改上面的程式,列印出線程 id 以便觀察並行任務的執行情況。

// final_answer_1.cpp
// compile with: /EHsc 

#include <ppltasks.h>
#include <iostream>
#include <thread>

using namespace concurrency;
using namespace std;

int main(int argc, char *argv[])
{
    cout << "Major thread id is: " << this_thread::get_id() << endl;

    task<int> final_answer([]
    {
        cout << "Thread id in task is:" << this_thread::get_id() << endl;
        return 42;
    });
    
    cout << "The final answer is: " << final_answer.get() << endl;
    
    return 0;
}

 

繼續編譯執行,得到輸出結果:

Major thread id is: 164824

Thread id in task is: 164824

The final answer is: 42

註意兩個線程 id 是相同的,很有些意外,任務是在主線程執行的而非預計的其他後臺工作線程。實際上這是 PPL 的優化策略造成的。

再修改下程式,在 task 對象構造完成後加一個 sleep 調用掛起當前線程一小段時間:

int main(int argc, char *argv[])
{
    cout << "Major thread id is: " << this_thread::get_id() << endl;

    task<int> final_answer([]
    {
        cout << "Thread id in task is:" << this_thread::get_id() << endl;
        return 42;
    });
    
    this_thread::sleep_for(chrono::milliseconds(1));

    cout << "The final answer is: " << final_answer.get() << endl;
    
    return 0;
}

 

這次輸出結果發生了變化:

Major thread id is: 173404

Thread id in task is: 185936

The final answer is: 42

PPL 使用了一個新的線程執行並行任務,實際上 PPL 是使用了線程池來執行被調度到的任務。

而在上一個程式中,由於沒有 sleep,也沒有其他耗時的代碼,執行到 task::get 方法時並行任務尚未被調度所以直接在當前線程執行該任務,這樣就節省了兩次線程切換的開銷

MSDN 中對 task::wait 方法的說明:

It is possible for wait to execute the task inline, if all of the tasks dependencies are satisfied, and it has not already been picked up for execution by a background worker.

task::get 方法的內部實現會先調用 task::wait 方法所以有同樣的效果。

 

本章小結:

1. task 類對象構造完成後即可被調度執行;

2. 並行有可能被優化在當前線程執行;

 

留一個問題,如果 task 對象構造後馬上析構,該並行任務是否會被調度執行呢?

 

本章代碼使用 visual studio community 2013 編譯調試通過。

本章參考文檔:

How to: Create a Task that Completes After a Delay 
task Class (Concurrency Runtime)


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

-Advertisement-
Play Games
更多相關文章
  • UBUNTU /boot/grub/grub.conf文件 ## DO NOT EDIT THIS FILE## It is automatically generated by grub-mkconfig using templates# from /etc/grub.d and settings ...
  • 1,linux安裝網路自動配置: 2,linux硬碟分配 1,/boot 用來存放與 Linux 系統啟動有關的程式,比如啟動引導裝載程式等,建議大小為 100-200MB 。 2,swap 實現虛擬記憶體,建議大小是物理記憶體的 1~2 倍。 3,/ 使用全部的磁碟空間 4,linux最小化安裝 英文 ...
  • 在centos上使用別名和是用密鑰登錄: vim /root/.ssh/config #輸入下列內容 Host * User root #以root登錄 ServerAliveInterval 60 #存活時間 #test1 Host q0 #登錄的別名 HostName 10.10.10.1 #i ...
  • 首先需要在centos系統下安裝好mysql,這個我已經安裝好了,這裡就不把過程貼出來了。 第一步:使用root用戶登錄到mysql資料庫: 第二步:創建一個mysql資料庫。當第一步登錄成功後,便會出現mysql命令:mysql> 第三步:在已經創建好的資料庫裡面創建一個簡單的數據表 第四步:為數 ...
  • 本文目錄: 4.1 文件系統的組成部分 4.2 文件系統的完整結構 4.3 Data Block 4.4 inode基礎知識 4.5 inode深入 4.6 單文件系統中文件操作的原理 4.7 多文件系統關聯 4.8 ext3文件系統的日誌功能 4.9 ext4文件系統 4.10 ext類文件系統的 ...
  • 公司有幾台美國的伺服器,平時也比較空閑,利用ssh搭了個隧道作為自己fq使用 辦公網路是一臺windows主機,不過上面裝了一個linux虛擬機,平時習慣使用虛擬機做開發機 辦公網路可以直接ssh通過內網地址鏈接美國伺服器 linux 虛擬機使用NAT模式上網ip:192.168.56.100 辦公 ...
  • http://blog.chinaunix.net/uid-7828352-id-4472376.html 寫的不錯,尤其是uboot向linux內核傳遞參數的過程寫的比較詳細。 ...
  • 由於我們有時候沒法預估或者說錯誤的盤符分區的時候,常常會導致我們後面的操作出現極大的不方便,這裡我就記錄下一個錯誤分區後對home和根分區存儲空間大小調整的整個過程!      ①查看我們現有機器的分區狀況 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...