0

我正在尝试从父子 (1-*) 实体集合模型中获取一个在标签中显示 2 个值的列表。

我有 3 个实体:

  1. [客户]:客户ID,姓名,地址,...
  2. [订单]:OrderId、OrderDate、EmployeeId、Total、...
  3. [ OrderStatus ]:OrderStatusId、StatusLevel、StatusDate、...

一个客户可以有很多订单,而一个订单可以有很多订单状态,即 [客户] 1--* [订单] 1-- * [订单状态]

给定一个 CustomerId,我想获取该订单的所有订单(仅 OrderId)和最新的(MAX?)OrderStatus.StatusDate。

我已经尝试了几次尝试,但似乎可以得到我想要的结果。

private IQueryable<Customer> GetOrderData(string customerId)
{
     var ordersWithLatestStatusDate = Context.Customers

     // Note: I am not sure if I should add the .Expand() extension methods here for the other two entity collections since I want these queries to be as performant as possible and since I am projecting below (only need to display 2 fields for each record in the IQueryable<T>, but thinking I should now after some contemplation.

     .Where(x => x.CustomerId == SelectedCustomer.CustomerId)
     .Select(x => new Custom 
     {
         CustomerId = x.CustomerId,
         ...
         // I would like to project my Child and GrandChild Collections, i.e. Orders and OrderStatuses here but don't know how to do that. I learned that by projecting, one does not need to "Include/Expand" these extension methods.  
     });
     return ordersWithLatestStatusDate ;
}

---- 更新 1 ----

在 User: lazyberezovsky的出色解决方案之后,我尝试了以下方法:

var query = Context.Customers
            .Where(c => c.CustomerId == SelectedCustomer.CustomerId)
            .Select(o => new Customer 
            { 
                Name = c.Name, 
                LatestOrderDate = o.OrderStatus.Max(s => s.StatusDate) 
            });

在我最初发布的仓促中,我没有正确粘贴所有内容,因为它主要来自内存并且当时没有确切的代码供参考。我的方法是一个强类型的 IQueryabled,我需要它返回一个类型为 T 的项目的集合,因为我必须通过一个严格的 API 中的约束,该 API 有一个 IQueryable 查询作为其参数之一。我知道我可以使用扩展方法 .Expand() 和/或 .Select() 添加其他实体/属性。人们会注意到我上面最新的更新查询在 .Select() 中添加了一个“新客户”,它曾经是匿名的。我很肯定这就是为什么查询失败的原因 b/c 由于 LatestOrderDate 不是客户在服务器级别的属性,它不能变成有效的 Uri。供参考,在看到下面的第一个答案后,我使用简单的 { get; 将该属性添加到我的客户端 Customer 类中。放; }。因此,鉴于此,我能否以某种方式仍然拥有一个 Customer 集合,只从 2 个不同的实体中带回这 2 个字段?下面的解决方案看起来如此有前途和巧妙!

---- 结束更新 1 ----

仅供参考,我使用的技术是 OData (WCF)、Silverlight、C#。

任何提示/链接将不胜感激。

4

2 回答 2

2

这将为您提供{ OrderId, LatestDate }对象列表

var query = Context.Customers
                   .Where(c => c.CustomerId == SelectedCustomer.CustomerId)
                   .SelectMany(c => c.Orders)
                   .Select(o => new { 
                              OrderId = o.OrderId, 
                              LatestDate = o.Statuses.Max(s => s.StatusDate) });

               .

UPDATE构造内存中的对象

var query = Context.Customers
                   .Where(c => c.CustomerId == SelectedCustomer.CustomerId)
                   .SelectMany(c => c.Orders)
                   .AsEnumerable() // goes in-memory
                   .Select(o => new { 
                              OrderId = o.OrderId, 
                              LatestDate = o.Statuses.Max(s => s.StatusDate) });

分组也可以在这里有所帮助。

于 2012-11-07T11:31:09.270 回答
0

如果我没看错,您需要一个 Customer 实体,然后是从其 Orders 属性计算的单个值。目前 OData 不支持此功能。OData 不支持查询中的计算值。所以投影中没有表达式,没有聚合等等。

不幸的是,即使有两个查询,这目前也是不可能的,因为 OData 不支持任何表达 MAX 功能的方式。

如果您可以控制服务,则可以编写服务器端功能/服务操作来执行此类查询。

于 2012-11-08T09:34:20.413 回答