Contract-First 工具
服務合約通常必須從現有的服務來建立。 在 .NET Framework 4.5 和更新版本中,可以使用合約優先工具自動從現有的服務建立資料合約類別。 若要使用合約優先工具,必須將 XML 結構描述定義 (XSD) 檔案下載至本機,這個工具無法透過 HTTP 匯入遠端資料合約。
合約優先工具已整合至 Visual Studio 2012 中做為建置工作。 每當建置專案時就會建立建置工作所產生的程式碼檔案,讓專案可以輕鬆採用基礎服務合約中的變更。
合約優先工具可以匯入的結構描述型別包括下列各項:
<xsd:complexType>
<xsd:simpleType>
</xsd:simpleType>
</xsd:complexType>
如果這些型別是基本型別 (例如 Int16
或 String
) 則不產生簡單型別,如果是 Collection
型別則不產生複雜類型。 如果這些型別是其他 xsd:complexType
的一部分,也不會產生型別。 在所有這些情況下,型別會改為參考專案中的現有型別。
將資料合約加入至專案
在使用合約優先工具之前,必須先將服務合約 (XSD) 加入至專案。 為了便於說明本概觀,將會使用下列合約來解釋合約優先功能。 這項服務定義是 Bing 的搜尋應用程式開發介面所使用之服務合約的一個小型子集合。
<?xml version="1.0" encoding="utf-8"?>
<xs:schema id="ServiceSchema"
targetNamespace="http://tempuri.org/ServiceSchema.xsd"
elementFormDefault="qualified"
xmlns="http://tempuri.org/ServiceSchema.xsd"
xmlns:mstns="http://tempuri.org/ServiceSchema.xsd"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
>
<xs:complexType name="SearchRequest">
<xs:sequence>
<xs:element minOccurs="0" maxOccurs="1" name="Version" type="xs:string" default="2.2" />
<xs:element minOccurs="0" maxOccurs="1" name="Market" type="xs:string" />
<xs:element minOccurs="0" maxOccurs="1" name="UILanguage" type="xs:string" />
<xs:element minOccurs="1" maxOccurs="1" name="Query" type="xs:string" />
<xs:element minOccurs="1" maxOccurs="1" name="AppId" type="xs:string" />
<xs:element minOccurs="0" maxOccurs="1" name="Latitude" type="xs:double" />
<xs:element minOccurs="0" maxOccurs="1" name="Longitude" type="xs:double" />
<xs:element minOccurs="0" maxOccurs="1" name="Radius" type="xs:double" />
</xs:sequence>
</xs:complexType>
<xs:simpleType name="WebSearchOption">
<xs:restriction base="xs:string">
<xs:enumeration value="DisableHostCollapsing" />
<xs:enumeration value="DisableQueryAlterations" />
</xs:restriction>
</xs:simpleType>
</xs:schema>
若要將上述服務合約加入至專案,請以滑鼠右鍵按一下專案並選取 [新增]。 從 [範本] 對話方塊的 [WCF] 窗格中選取 [結構描述定義],並命名新的檔案 SampleContract.xsd。 複製上面的程式碼,將其貼入新檔案的程式碼檢視中。
設定合約優先選項
合約優先選項可以在 WCF 專案的 [屬性] 功能表進行設定。 若要啟用合約優先開發,請選取專案屬性視窗之 WCF 頁面中的 [將 XSD 啟用為型別定義語言] 核取方塊。
若要設定進階屬性,請按一下 [進階] 按鈕。
您可以設定下列適用於從合約產生程式碼的進階設定。 這些設定只能針對專案中的所有檔案進行,目前尚無法對個別檔案進行設定。
序列化程式模式:這個設定決定使用哪個序列化程式來讀取服務合約檔。 選取 [XML 序列化程式] 時會停用 [集合型別] 和 [重複使用型別] 選項。 這些選項僅適用於 [資料合約序列化程式]。
重複使用型別:這個設定指定哪些程式庫會用於型別重複使用。 這個設定只有在 [序列化程式模式] 設定為 [資料合約序列化程式] 時才適用。
集合型別:這個設定指定集合資料型別要使用完整限定或組件限定型別。 這個設定只有在 [序列化程式模式] 設定為 [資料合約序列化程式] 時才適用。
字典型別:這個設定指定集合資料型別要使用完整限定或組件限定型別。
EnableDataBinding:這個設定指定是否要實作所有資料型別上的 INotifyPropertyChanged 介面以實作資料繫結。
ExcludedTypes:這個設定指定要從參考組件中排除之完整限定或組件限定型別的清單。 這個設定只有在 [序列化程式模式] 設定為 [資料合約序列化程式] 時才適用。
GenerateInternalTypes:這個設定指定是否產生標記為內部的類別。 這個設定只有在 [序列化程式模式] 設定為 [資料合約序列化程式] 時才適用。
GenerateSerializableTypes:這個設定指定是否產生具有 SerializableAttribute 屬性的類別。 這個設定只有在 [序列化程式模式] 設定為 [資料合約序列化程式] 時才適用。
ImportXMLTypes:這個設定指定是否將資料合約序列化程式設定為套用 SerializableAttribute 屬性至沒有 DataContractAttribute 屬性的類別。 這個設定只有在 [序列化程式模式] 設定為 [資料合約序列化程式] 時才適用。
SupportFx35TypedDataSets:這個設定指定是否要為針對 .NET Framework 3.5 所建立的具型別資料集提供其他功能。 當 [序列化程式模式] 設定為 [XML 序列化程式] 時,只要這個值設定為 True,就會將 TypedDataSetSchemaImporterExtensionFx35 擴充加入至 XML 結構描述匯入工具。 當 [序列化程式模式] 設定為 [資料合約序列化程式] 時,只要這個值設定為 False,就會從 [參考] 中排除 DateTimeOffset 型別,使 DateTimeOffset 永遠針對舊版 Framework 而產生。
InputXsdFiles:這個設定指定輸入檔清單。 每個檔案都必須包含有效的 XML 結構描述。
語言:這個設定指定產生的合約程式碼的語言。 這個設定必須可以由 CodeDomProvider 辨識。
NamespaceMappings:這個設定指定從 XSD 目標命名空間到 CLR 命名空間的對應。 每個對應都要使用下列格式:
"Schema Namespace, CLR Namespace"
XML 序列化程式只接受一個使用下列格式的對應:
"*, CLR Namespace"
OutputDirectory:這個設定指定產生程式碼檔案所在的目錄。
建置專案時,將會使用這些設定從服務合約檔案產生服務合約型別。
使用合約優先開發
將服務合約加入至專案並確認組建設定之後,按下 F6 以建置專案。 服務合約中定義的型別接著就可以在專案中使用。
若要使用服務合約中定義的型別,請在目前命名空間之下加入 ContractTypes
的參考:
using MyProjectNamespace.ContractTypes;
服務合約中定義的型別接著就可以在專案中解析,如下所示:
工具所產生的型別會建立在 GeneratedXSDTypes.cs 檔案中。 預設會在 <專案目錄>/obj/<組建組態>/XSDGeneratedCode/ 目錄中建立這個檔案。 本文開頭的範例結構描述轉換如下:
//------------------------------------------------------------------------------
// <auto-generated>
// This code was generated by a tool.
// Runtime Version:4.0.30319.17330
//
// Changes to this file may cause incorrect behavior and will be lost if
// the code is regenerated.
// </auto-generated>
//------------------------------------------------------------------------------
namespace TestXSD3.ContractTypes
{
using System.Xml.Serialization;
/// <remarks/>
[System.CodeDom.Compiler.GeneratedCodeAttribute("System.Xml", "4.0.30319.17330")]
[System.SerializableAttribute()]
[System.Diagnostics.DebuggerStepThroughAttribute()]
[System.ComponentModel.DesignerCategoryAttribute("code")]
[System.Xml.Serialization.XmlTypeAttribute(Namespace="http://tempuri.org/ServiceSchema.xsd")]
[System.Xml.Serialization.XmlRootAttribute(Namespace="http://tempuri.org/ServiceSchema.xsd", IsNullable=true)]
public partial class SearchRequest
{
private string versionField;
private string marketField;
private string uILanguageField;
private string queryField;
private string appIdField;
private double latitudeField;
private bool latitudeFieldSpecified;
private double longitudeField;
private bool longitudeFieldSpecified;
private double radiusField;
private bool radiusFieldSpecified;
public SearchRequest()
{
this.versionField = "2.2";
}
/// <remarks/>
[System.ComponentModel.DefaultValueAttribute("2.2")]
public string Version
{
get
{
return this.versionField;
}
set
{
this.versionField = value;
}
}
/// <remarks/>
public string Market
{
get
{
return this.marketField;
}
set
{
this.marketField = value;
}
}
/// <remarks/>
public string UILanguage
{
get
{
return this.uILanguageField;
}
set
{
this.uILanguageField = value;
}
}
/// <remarks/>
public string Query
{
get
{
return this.queryField;
}
set
{
this.queryField = value;
}
}
/// <remarks/>
public string AppId
{
get
{
return this.appIdField;
}
set
{
this.appIdField = value;
}
}
/// <remarks/>
public double Latitude
{
get
{
return this.latitudeField;
}
set
{
this.latitudeField = value;
}
}
/// <remarks/>
[System.Xml.Serialization.XmlIgnoreAttribute()]
public bool LatitudeSpecified
{
get
{
return this.latitudeFieldSpecified;
}
set
{
this.latitudeFieldSpecified = value;
}
}
/// <remarks/>
public double Longitude
{
get
{
return this.longitudeField;
}
set
{
this.longitudeField = value;
}
}
/// <remarks/>
[System.Xml.Serialization.XmlIgnoreAttribute()]
public bool LongitudeSpecified
{
get
{
return this.longitudeFieldSpecified;
}
set
{
this.longitudeFieldSpecified = value;
}
}
/// <remarks/>
public double Radius
{
get
{
return this.radiusField;
}
set
{
this.radiusField = value;
}
}
/// <remarks/>
[System.Xml.Serialization.XmlIgnoreAttribute()]
public bool RadiusSpecified
{
get
{
return this.radiusFieldSpecified;
}
set
{
this.radiusFieldSpecified = value;
}
}
}
/// <remarks/>
[System.CodeDom.Compiler.GeneratedCodeAttribute("System.Xml", "4.0.30319.17330")]
[System.SerializableAttribute()]
[System.Xml.Serialization.XmlTypeAttribute(Namespace="http://tempuri.org/ServiceSchema.xsd")]
[System.Xml.Serialization.XmlRootAttribute(Namespace="http://tempuri.org/ServiceSchema.xsd", IsNullable=false)]
public enum WebSearchOption
{
/// <remarks/>
DisableHostCollapsing,
/// <remarks/>
DisableQueryAlterations,
}
}
錯誤和警告
剖析 XSD 結構描述時發生的錯誤和警告將會顯示為建置錯誤和警告。
介面繼承
不能同時使用介面繼承與合約優先開發;這與介面在其他作業中的運作方式一致。 若要使用繼承基底介面的介面,請使用兩個單獨的端點。 第一個端點使用繼承的合約,第二個端點實作基底介面。