契約Contract(ServiceContract、OperationContract、DataContract、ServiceKnownType和DataMember)
介紹
WCF(Windows Communication Foundation) - 契約(Contract):服務契約(ServiceContract),操作契約(OperationContract),數據契約(DataContract),服務已知類型(ServiceKnownType),數據成員(DataMember)。
示例
1、服務
IPersonManager.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.ServiceModel;
using System.Runtime.Serialization;
namespace WCF.ServiceLib.Contract
{
/**//// <summary>
/// 人員管理接口
/// </summary>
// Namespace - 服務契約的命名空間
// Name - 服務契約的名稱(會對應到相關的wsdl,默認情況下本例為接口名“IPersonManager”)
// ConfigurationName - 服務契約在宿主中所配置的服務名稱(默認情況下本例為類的全名“WCF.ServiceLib.Contract.IPersonManager”)
[ServiceContract(Namespace = "http://webabcd.cnblogs.com", Name = "IPersonManager", ConfigurationName = "ConfigurationNameTest")]
// 服務已知類型 - Student(數據契約)繼承自Person(數據契約),要指定Student為已知類型,其才會被序列化
[ServiceKnownType(typeof(Student))]
public interface IPersonManager
{
/**//// <summary>
/// 獲取某人的姓名
/// </summary>
/// <param name="p">Person對象</param>
/// <returns></returns>
// Name - 操作契約的名稱(會對應到相關的wsdl,默認情況下本例為方法名“GetName”)
[OperationContract(Name="GetPersonName")]
string GetName([MessageParameter(Name = "person")] Person p);
}
}
PersonManager.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.ServiceModel;
using System.Runtime.Serialization;
namespace WCF.ServiceLib.Contract
{
/**//// <summary>
/// 人員管理類
/// </summary>
public class PersonManager : IPersonManager
{
/**//// <summary>
/// 獲取某人的姓名
/// </summary>
/// <param name="p">Person對象</param>
/// <returns></returns>
public string GetName(Person p)
{
return "Name: " + p.Name;
}
}
}
Person.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.ServiceModel;
using System.Runtime.Serialization;
namespace WCF.ServiceLib.Contract
{
/**//// <summary>
/// Person的實體類
/// </summary>
// Name - 數據契約的名稱(會對應到相關的wsdl,默認情況下本例為類名“Person”)
[DataContract(Name = "PersonModel")]
public class Person
{
/**//// <summary>
/// Person的實體類的Age屬性
/// </summary>
// Name - 數據成員的名稱(會對應到相關的wsdl,默認情況下本例為屬性名“Age”)
// IsRequired - 該值指示序列化引擎該成員在讀取或反序列化時必須存在
// Order - 數據成員在相關的wsdl中的順序
// EmitDefaultValue - 如果應該在序列化流中生成成員的默認值,則為 true,否則為 false,默認值為 true
[DataMember(Name = "PersonAge", IsRequired = false, Order = 1)]
public int Age { get; set; }
/**//// <summary>
/// Person的實體類的Name屬性
/// </summary>
// Name - 數據成員的名稱(會對應到相關的wsdl,默認情況下本例為屬性名“Name”)
// IsRequired - 該值指示序列化引擎該成員在讀取或反序列化時必須存在
// Order - 數據成員在相關的wsdl中的順序
// EmitDefaultValue - 如果應該在序列化流中生成成員的默認值,則為 true,否則為 false,默認值為 true
[DataMember(Name = "PersonName", IsRequired = false, Order = 0)]
public string Name { get; set; }
}
}
Student.cs
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.ServiceModel;
using System.Runtime.Serialization;
namespace WCF.ServiceLib.Contract
{
/**//// <summary>
/// Student的實體類
/// </summary>
// Name - 數據契約的名稱(會對應到相關的wsdl,默認情況下本例為類名“Student”)
[DataContract(Name = "StudentModel")]
public class Student : Person
{
/**//// <summary>
/// Student的實體類的School屬性
/// </summary>
// Name - 數據成員的名稱(會對應到相關的wsdl,默認情況下本例為屬性名“School”)
// IsRequired - 該值指示序列化引擎該成員在讀取或反序列化時必須存在
// Order - 數據成員在相關的wsdl中的順序
// EmitDefaultValue - 如果應該在序列化流中生成成員的默認值,則為 true,否則為 false,默認值為 true
[DataMember(Name = "School", IsRequired = false, Order = 0)]
public string School { get; set; }
}
}
2、宿主
PersonManager.svc
<%@ ServiceHost Language="C#" Debug="true" Service="WCF.ServiceLib.Contract.PersonManager" %>
Web.config
<?xml version="1.0"?> <configuration> <system.serviceModel> <behaviors> <serviceBehaviors> <behavior name="ContractBehavior"> <!--httpGetEnabled - 使用get方式提供服務--> <serviceMetadata httpGetEnabled="true" /> </behavior> </serviceBehaviors> </behaviors> <services> <!--name - 提供服務的類名--> <!--behaviorConfiguration - 指定相關的行為配置--> <service name="WCF.ServiceLib.Contract.PersonManager" behaviorConfiguration="ContractBehavior"> <!--address - 服務地址--> <!--binding - 通信方式--> <!--contract - 服務契約--> <endpoint address="" binding="basicHttpBinding" contract="ConfigurationNameTest" /> </service> </services> </system.serviceModel> </configuration>
3、客戶端
PersonManager.aspx
<%@ Page Language="C#" MasterPageFile="~/Site.master" AutoEventWireup="true" CodeFile="PersonManager.aspx.cs" Inherits="Contract_PersonManager" Title="契約(ServiceContract、OperationContract、DataContract、ServiceKnownType和DataMember)" %> <asp:Content ID="Content1" ContentPlaceHolderID="head" runat="Server"> </asp:Content> <asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" runat="Server"> <asp:TextBox ID="txtName" runat="server" Text="webabcd" /> <asp:Button ID="btnGetName" runat="server" Text="GetName" onclick="btnGetName_Click" /> </asp:Content>
PersonManager.aspx.cs
using System;
using System.Collections;
using System.Configuration;
using System.Data;
using System.Linq;
using System.Web;
using System.Web.Security;
using System.Web.UI;
using System.Web.UI.HtmlControls;
using System.Web.UI.WebControls;
using System.Web.UI.WebControls.WebParts;
using System.Xml.Linq;
public partial class Contract_PersonManager : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
}
protected void btnGetName_Click(object sender, EventArgs e)
{
// Contract.IPersonManager pm = new Contract.PersonManagerClient();
Contract.PersonManagerClient proxy = new Contract.PersonManagerClient();
Contract.StudentModel sm = new Contract.StudentModel() { PersonName = txtName.Text };
Page.ClientScript.RegisterStartupScript(
this.GetType(),
"js",
string.Format("alert('{0}')", proxy.GetPersonName(sm)),
true);
proxy.Close();
}
}
Web.config
<?xml version="1.0"?> <configuration> <system.serviceModel> <client> <!--address - 服務地址--> <!--binding - 通信方式--> <!--contract - 服務契約--> <endpoint address="http://localhost:3502/ServiceHost/Contract/PersonManager.svc" binding="basicHttpBinding" contract="Contract.IPersonManager" /> </client> </system.serviceModel> </configuration>
運行結果:
單擊"btnGetName"後彈出提示框,顯示"Name: webabcd"
OK