共用方式為


累加建置

累加建置是最佳化的建置,如此才不會執行擁有最新輸出檔 (相對於其對應的輸入檔) 的目標。目標項目 (Element) 可以有 Inputs 屬性 (Attribute) 和 Outputs 屬性,前者表示目標預期做為輸入的項目 (Item),後者則表示目標會產生為輸出的項目 (Item)。MSBuild 會試著在這些屬性值之間尋找一對一的對應。如果有一對一的對應存在,MSBuild 會比較每一個輸入項目的時間戳記與其對應輸出項目的時間戳記。沒有一對一對應的輸出檔會與所有輸入檔進行比較。如果項目的輸出檔與輸入檔同齡或是前者較舊,該項目則可視為最新狀態。

如果所有輸出項目都是最新的,MSBuild 便會略過該目標。目標的這個「累加建置」(Incremental Build) 可以大幅改善建置速度。如果只有部分檔案是最新的,MSBuild 則會執行該目標,但是略過最新的項目,由此讓所有項目成為最新狀態。這個程序稱為「部分累加建置」(Partial Incremental Build)。

一對一對應通常是由項目轉換所產生。如需詳細資訊,請參閱 MSBuild 轉換

請參考下列目標:

<Target Name="Backup" Inputs="@(Compile)" 
    Outputs="@(Compile->'$(BackupFolder)%(Identity).bak')">
    <Copy SourceFiles="@(Compile)" DestinationFiles=
        "@(Compile->'$(BackupFolder)%(Identity).bak')" />
</Target>

Compile 項目類型所代表的檔案集會複製到備份目錄。備份檔的副檔名為 .bak。如果在執行備份目標之後,並未刪除或修改 Compile 項目類型所代表的檔案或是對應的備份檔,則會在後續的建置中略過此備份目標。

輸出推斷

MSBuild 會比較目標的 Inputs 和 Outputs 屬性,以決定是否必須執行目標。理想上,不論是否執行關聯的目標,在完成累加建置之後存在的檔案集都應該維持不變。因為工作所建立或修改的屬性 (Property) 和項目可能會影響組建,所以即使略過了會造成影響的目標,MSBuild 仍必須推斷其值。這稱為「輸出推斷」(Output Inference)。

具有下列三種情況:

  • 目標的 Condition 屬性 (Attribute) 評估為 false。在此情況下,目標並未執行,而且對組建沒有任何影響。

  • 目標具有最新的輸出,並予以執行以產生最新輸出。

  • 目標沒有最新的輸出,並予以略過。MSBuild 會評估目標並且變更項目和屬性 (Property),彷彿已執行過目標一樣。

若要支援累加編譯,工作必須確保任何 Output 項目 (Element) 的 TaskParameter 屬性 (Attribute) 值等於工作輸入參數。以下是一些範例:

<CreateProperty Value="123">
    <Output PropertyName="Easy" TaskParameter="Value" />
</CreateProperty>

這會建立 Easy 屬性 (Property),而不論是執行或略過目標,該屬性的值都是 "123"。

<CreateItem Include="a.cs;b.cs">
    <Output ItemName="Simple" TaskParameter="Include" />
</CreateItem>

這會建立 Simple 項目類型,而不論是執行或略過目標,該類型都具有兩個項目:"a.cs" 和 "b.cs"。

在 MSBuild 3.5 中,系統會自動對目標中的項目和屬性群組執行輸出推斷。目標不需要 CreateItem 工作,應予以避免。此外,CreateProperty 工作只能用於目標,以判斷目標是否已經執行。

判斷目標是否已經執行

由於執行輸出推斷,所以您必須將 CreateProperty 工作加入至目標以檢查其屬性和項目,以便判斷目標是否已經執行。將 CreateProperty 工作加入至目標,並將 TaskParameter 為 "ValueSetByTask" 的 Output 項目提供給它。

<CreateProperty Value="true">
    <Output TaskParameter="ValueSetByTask" PropertyName="CompileRan" />
</CreateProperty>

這會建立 CompileRan 屬性並提供 true 值,但僅限於已經執行目標時。如果目標已經略過,則不會建立 CompileRan。

請參閱

概念

MSBuild 目標