並行執行階段概觀
本文件提供並行執行階段的總覽。 內容描述並行執行階段的優點、使用時機,以及其元件如何與彼此、與作業系統及應用程式互動。
區段
本文件包含下列章節:
並行運行時間實作歷程記錄
在 Visual Studio 2010 到 2013 中,並行運行時間會透過 msvcr120.dll 併入msvcr100.dll內。 Visual Studio 2015 中發生 UCRT 重構時,該 DLL 已重構為三個部分:
ucrtbase.dll – C API,隨附於 Windows 10 中,並透過 Windows Update 提供舊版服務-
vcruntime140.dll – 編譯程式支援函式和 EH 運行時間,透過 Visual Studio 隨附
concrt140.dll – 透過 Visual Studio 隨附的並行運行時間。 平行容器和演算法的必要專案,例如
concurrency::parallel_for
。 此外,STL 需要 Windows XP 上的這個 DLL 來提供同步處理基本類型,因為 Windows XP 沒有條件變數。
在 Visual Studio 2015 和更新版本中,並行執行階段工作排程器不再是 task 類別及 ppltasks.h 中相關類型適用的排程器。 那些類型現在使用 Windows 執行緒集區以獲得更佳的效能,以及與 Windows 同步處理原始物件的互通性。
為什麼並行運行時間很重要
並行執行階段為同時執行的應用程式與應用程式元件提供一致性和可預測性。 並行運行時間的優點有兩個範例,包括 合作式工作排程 和 合作封鎖。
並行執行階段會使用實作工作竊取演算法的合作式工作排程器,有效率地在運算資源之間分散工作。 例如,假設有個應用程式具有兩個執行緒,且都是由相同的執行階段管理。 如果一個執行緒完成其排定工作,它便可以從另一個執行緒卸載工作。 這項機制可在應用程式的整體工作負載之間取得平衡。
並行執行階段也會提供同步處理原始物件,使用合作式封鎖來同步處理對資源的存取。 例如,假設有一項工作必須具有對共用資源的獨佔存取權。 藉由合作方式封鎖,在第一項工作等候資源時,執行階段便可以使用剩餘的配量執行另一項工作。 這項機制會提升運算資源的最大使用量。
[靠上]
架構
並行執行階段分成四個元件:平行模式程式庫 (PPL)、非同步代理程式程式庫、工作排程器和資源管理員。 這些元件位於作業系統和應用程式之間。 下圖顯示並行執行階段元件在作業系統和應用程式之間互動的方式:
並行運行時間架構
重要
當您在 ppltasks.h 中使用工作類別或其他類型時,工作排程器和 Resource Manager 元件無法從 通用 Windows 平台 (UWP) 應用程式取得。
並行運行時間高度 可組合,也就是說,您可以結合現有的功能來執行更多作業。 並行執行階段會組合來自較低層級元件的許多功能,例如平行演算法。
並行執行階段也會提供同步處理原始物件,使用合作式封鎖來同步處理對資源的存取。 如需這些同步處理基本類型的詳細資訊,請參閱 同步處理數據結構。
下列各節提供每個元件所提供功能及其使用時機的簡短總覽。
平行模式程式庫
平行模式程式庫 (PPL) 提供一般用途的容器和演算法來執行細部平行處理原則。 PPL 藉由提供平行演算法,在集合上或跨運算資源數據集散發計算,來啟用 命令式數據平行 處理原則。 它也會藉由提供工作物件,將多個獨立作業分散到運算資源,藉此啟用 工作平行 處理原則。
如果您擁有可受益於平行執行的本機計算,請使用平行模式程式庫。 例如,您可以使用 concurrency::p arallel_for 演算法來轉換現有的 for
迴圈,以平行方式運作。
如需平行模式連結庫的詳細資訊,請參閱 平行模式連結庫 (PPL) 。
非同步代理程式程式庫
異步代理程式連結庫(或只提供 代理程序連結庫)同時提供動作專案型程式設計模型和訊息傳遞介面,以用於粗細數據流和管道處理工作。 非同步代理程式可讓您在其他元件等候資料時執行工作,以便對延遲進行具生產力的利用。
當您有多個實體以非同步方式彼此通訊時,請使用代理程式庫。 例如,您可以建立代理程式,從檔案或網路連線讀取資料,然後使用訊息傳遞介面將該資料傳送至另一個代理程式。
如需 Agents 連結庫的詳細資訊,請參閱 異步代理程式連結庫。
工作排程器
工作排程器會在執行階段排定並協調工作。 工作排程器是合作式的,並且使用工作竊取演算法來達到處理資源的最大使用率。
並行執行階段提供預設排程器,讓您不需要管理基礎結構的細節。 不過,為了滿足應用程式的品質需求,您也可以提供自己的排程原則或建立特定排程器與特定工作的關聯。
如需工作排程器的詳細資訊,請參閱 工作排程器。
Resource Manager
資源管理員的角色是管理運算資源,例如處理器和記憶體。 資源管理員會藉由將資源指派到最能發揮效果之處,在工作負載於執行階段變更時做出回應。
資源管理員是運算資源的抽象概念,並且主要與工作排程器互動。 雖然您可以使用資源管理員來微調程式庫和應用程式的效能,但您通常會使用平行模式程式庫、代理程式庫和工作排程器所提供的功能。 這些程式庫會使用資源管理員,在工作負載變更時,動態地重新平衡資源。
[靠上]
C++ Lambda 運算式
並行執行階段所定義的許多類型和演算法會實作為 C++ 範本。 其中有些類型和演算法需要執行工作的常式做為參數。 這個參數可以是 Lambda 函式、函式物件或函式指標。 這些實體也稱為 工作函 式或 工作例程。
Lambda 運算式是一項重要的新 Visual C++ 語言功能,因為此運算式提供簡潔的方式來定義平行處理用的工作函式。 函式物件和函式指標可讓您使用並行執行階段搭配現有的程式碼。 不過,我們建議您在撰寫新程式碼時使用 Lambda 運算式,因為它們提供安全性和產能優勢。
下列範例會比較多個並行呼叫 中 Lambda 函式、函式物件和函式指標的語法::p arallel_for_each 演算法。 每次呼叫 parallel_for_each
都會使用不同的技術來計算 std::array 物件中每個專案的平方。
// comparing-work-functions.cpp
// compile with: /EHsc
#include <ppl.h>
#include <array>
#include <iostream>
using namespace concurrency;
using namespace std;
// Function object (functor) class that computes the square of its input.
template<class Ty>
class SquareFunctor
{
public:
void operator()(Ty& n) const
{
n *= n;
}
};
// Function that computes the square of its input.
template<class Ty>
void square_function(Ty& n)
{
n *= n;
}
int wmain()
{
// Create an array object that contains 5 values.
array<int, 5> values = { 1, 2, 3, 4, 5 };
// Use a lambda function, a function object, and a function pointer to
// compute the square of each element of the array in parallel.
// Use a lambda function to square each element.
parallel_for_each(begin(values), end(values), [](int& n){n *= n;});
// Use a function object (functor) to square each element.
parallel_for_each(begin(values), end(values), SquareFunctor<int>());
// Use a function pointer to square each element.
parallel_for_each(begin(values), end(values), &square_function<int>);
// Print each element of the array to the console.
for_each(begin(values), end(values), [](int& n) {
wcout << n << endl;
});
}
輸出
1
256
6561
65536
390625
如需C++中 Lambda 函式的詳細資訊,請參閱 Lambda 表達式。
[靠上]
需求
下表顯示與每個並行執行階段元件關聯的標頭檔:
元件 | 標頭檔 |
---|---|
平行模式程式庫 (PPL) | ppl.h concurrent_queue.h concurrent_vector.h |
非同步代理程式程式庫 | agents.h |
工作排程器 | concrt.h |
Resource Manager | concrtrm.h |
並行運行時間會在並行命名空間中宣告。 (您也可以使用 並行存取,這是這個命名空間的別名。命名空間 concurrency::details
支援並行運行時間架構,並不適合直接從您的程式代碼使用。
並行執行階段被提供為 C 執行階段程式庫 (CRT) 的一部分。 如需如何建置使用CRT的應用程式的詳細資訊,請參閱 CRT連結庫功能。
[靠上]