先创建一个 WCF Service Application

删掉默认的 Service1.svc 和 IService.cs,添加一个 WcfDataService.svc 项目

查看 SVC 文件的 Markup,你会发现它加上 DataServiceHostFactory:
<%@ ServiceHost Language="C#" Factory="System.Data.Services.DataServiceHostFactory, System.Data.Services, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" Service="WcfCustomerDataService.WcfDataService1" %>
如果在开发非IIS Host应用的时候,则使用 WebServiceHost 来寄宿服务。详细参考:Hosting the Data Service (ADO.NET Data Services)
1.定义实体:实体里必须使用 DataServiceKeyAttribute 标注Key。否则DataService运行报错。[DataServiceKeyAttribute("OrderId")]
public class Order
{
public int OrderId { get; set; }
public string Customer { get; set; }
public IList<Item> Items { get; set; }
}
[DataServiceKeyAttribute("Product")]
public class Item
{
public string Product { get; set; }
public int Quantity { get; set; }
}
2. 定义数据模型:
(1) 在 DataModel 的静态构造方法中,添加了一些初始数据,在这里你可以任意扩展数据的来源。
(2) 定义了两个属性:Orders 和 Items, 它们返回 IQueryable<T> 的结果。
public class DataModel
{
#region Populate Service Data
static IList<Order> _orders;
static IList<Item> _items;
static DataModel()
{
_orders = new List<Order> {
new Order(){ OrderId=1, Customer = "Wendy Wu", Items = new List<Item>()},
new Order(){ OrderId=2, Customer = "John Gu", Items = new List<Item>()},
new Order(){ OrderId=3, Customer = "Balance Yao", Items = new List<Item>()}
};
_items = new List<Item> {
new Item(){ Product="Chang", Quantity = 4 },
new Item(){ Product="Aniseed Syrup", Quantity=5 },
new Item(){ Product="Toy", Quantity=7 },
new Item(){ Product="Car", Quantity=1 },
new Item(){ Product="Ball", Quantity=6}
};
_orders[0].Items.Add(_items[0]);
_orders[1].Items.Add(_items[1]);
_orders[1].Items.Add(_items[2]);
_orders[2].Items.Add(_items[3]);
_orders[2].Items.Add(_items[4]);
}
#endregion
public IQueryable<Order> Orders
{
get { return _orders.AsQueryable<Order>(); }
}
public IQueryable<Item> Items
{
get { return _items.AsQueryable<Item>(); }
}
}
3. 实现Service
public class WcfDataService1 : DataService<DataModel>
{
public static void InitializeService(DataServiceConfiguration config)
{
config.SetEntitySetAccessRule("*", EntitySetRights.All);
config.SetServiceOperationAccessRule("*", ServiceOperationRights.All);
config.DataServiceBehavior.MaxProtocolVersion = DataServiceProtocolVersion.V2;
}
}
运行,在浏览器里输入 http://localhost:50480/WcfDataService1.svc/Orders 就能看到结果:返回Atom协议的XML数据。
PS:Data Service 通过客户端Request的Accept 来决定返回格式。
比如我用 Fiddler 来查询,使用Accept: application/json 那么返回的就是Json格式的数据:
比如在上面的 WcfDataService1 里加入下面的方法:调用时按照OData协议写Query
[WebGet]
[SingleResult]
public Order TopOrder()
{
return this.CurrentDataSource.Orders
.OrderByDescending(o => o.Items.Sum(i => i.Quantity)).First();
}
Data Service 对服务操作有一些具体的限制 - 此方法只能接受 [in] 参数。如果对参数进行定义,则每个参数的类型必须为基元类型。
- 此方法必须返回 void、IEnumerable<T>、IQueryable<T>、T 或基元类 (如整数或字符串)。T 必须为一个类,此类表示数据服务将公开的数据模型中的某个实体类型。若要支持查询选项(如排序、分页和筛选),服务操作方法应返回 IQueryable<T>。
- 必须用 [WebGet] 或 [WebInvoke] 属性为此方法添加标注。[WebGet] 使调用方能够通过使用 GET 请求调用此方法;[WebInvoke] 使调用方能够通过使用 PUT、POST 或 DELETE 请求调用此方法。
- 可以用 SingleResultAttribute 为服务操作添加批注,指定此方法的返回值是一个实体而不是一个实体集。这一区别确定了生成的响应序列化。例如,当使用 AtomPub 序列化时,单个资源类型实例将表示为一个 entry 元素,而单个实例集将表示为一个 feed 元素。
创建一个Console工程,添加Service Reference 生成客户端代理,调用DataService:
static void Main(string[] args)
{
var svcUri = new Uri("http://localhost:50480/WcfDataService1.svc");
//Query
var ctx = new DataSvc.DataModel(svcUri);
var order = ctx.Orders.Where(o => o.OrderId == 3).First();
Console.WriteLine(order.Customer);
//Query by customer method 'TopOrder'
var topOrder = ctx.Execute<DataSvc.Order>(new Uri("/TopOrder", UriKind.Relative)).First();
Console.WriteLine(topOrder.Customer);
Console.Read();
}
参考:http://msdn.microsoft.com/zh-cn/library/dd723653.aspx
(2011/7/30 更新)
通过 QueryInterceptorAttribute 和 ChangeInterceptorAttribute 可以对请求进行拦截。
参考:http://msdn.microsoft.com/zh-cn/library/dd744837.aspx