10

假设我有一个看起来像这样的 OData 查询(我的实际查询要复杂得多):

Orders.Select(z => new { z.SubOrder.Addresses,
                         z.SubOrder.Cost,
                         z.SubOrder.SubOrderId, 
                         z.Sequence});

这工作正常。除了 Address 对象上有一个子对象(StateRef)。由于 StateRef 对 State 表进行了查找,因此它返回为 null。

为了说明,下面是地址对象 Address 的示例:

Address:
    string         Street 1
    string         Street 2
    StateRef       PrimaryState
    string         City
    // ... 42 other string attributes not shown ...

StateRef对象上面有状态的名称,但也有一些其他重要的状态属性(可能是状态鸟?)

所以,我想知道的是,我现在是否必须为包含所有 46 个属性的 z.SubOrder.Addresses 创建一个“子投影”,以便我可以访问该PrimaryState项目?(我希望不是

除了更多的编码之外,这还意味着我必须使用匿名类型。这使得我的映射必须手动(而不是使用 AutoMapper)。

那么我正在寻找一种在投影内“扩展”StateRef 的方法?

像这样的东西:

Orders.Select(z => new { z.SubOrder.Addresses.Expand("PrimaryState"),
                         z.SubOrder.Cost,        ^
                         z.SubOrder.SubOrderId,  |
                         z.Sequence});           |
                                                 |
// This is not allowed by the compiler ----------+

尝试这个会给出这个错误:

无效的匿名类型成员声明符。必须使用成员分配、简单名称或成员访问来声明匿名类型成员。

更新: 这是一个示例查询来说明我在问什么:

Users.Take(10).Select(x=>new { x.Id, x.Reputation, x.Comments})

针对“ data.stackexchange.com/stackoverflow/atom ”运行它。您将看到 Comments 有一个返回 null 的 Post 对象。

我需要它来返回其中的值。

注意:我知道我可以手动将它们全部输入到“子”投影中。阅读上面的内容,了解我不想要的原因。

4

5 回答 5

7

这样做当然是可能的。对于概念证明,请尝试执行以下操作:

var uri = new Uri( "http://data.stackexchange.com/stackoverflow/atom/Users()?$top=10&$expand=Comments/Post&$select=Id,Reputation,Comments/" );
entities.Execute<User>( uri, "GET", false ).Select( x => new { x.Id, x.Reputation, x.Comments } );

expand的正确用法是这样的:

entities.Users.Expand( "Comments/Post" ).Take( 10 ).ToArray();

我不知道为什么图书馆的作者决定禁止使用带有投影的扩展,但正如上面的概念证明所示,这样做当然是可能的。

如果您不介意接收整个用户并在此之后进行投影,则可以使用第二个示例。否则,您可以编写自己的助手,它将从第一个示例中生成 URI,执行它们,然后添加投影。

于 2013-02-15T11:43:16.037 回答
2

您不必创建列出所有 46 个属性的子投影,例如

(from u in Users
 select new 
    {
    u.Id, u.Reputation,Comments = ( from c in u.Comments 
                                    select new YourClass {comment = c, 
                                                          post= c.Post})
    }
)
.Take(10)


.....


public Class YourClass
    {
    public Comment comment {get;  set;}
    public Post post {get;set;}
    }

不完全是我想象的你想要的对象图。

除此之外,人们可能会花费大量时间来尝试编写将生成正确 OData 查询的 LinQ 表达式,我们发现使用 Expand、Filter、Select 属性等创建自己的 OData 查询类会更省时. 即直接编写 OData 查询,而不是尝试制作 LinQ 查询。

于 2013-02-11T17:06:46.487 回答
2

你可以做,

Users.Take(10).Select(x=>new { Id = x.Id, Reputation = x.Reputation, Comments = x.Comments.Select(c =>c.Id) })

这将导致请求,

https://data.stackexchange.com/stackoverflow/atom/Users()?$top=10&$expand=Comments&$select=Id,Reputation,Comments/Id

于 2013-02-15T18:27:38.753 回答
1

您可以从子对象中选择特定属性。

例如,在 StackOverflow 的示例中,我可以在 LINQPad 中成功执行以下查询。

Users
.Take (10)
.Select(x => new {x.Id, x.Reputation, CommentsText = x.Comments.Select(c => c.Text)})

在您的情况下,您可以编写如下查询:

Orders.Select(z => new { StateName = z.SubOrder.Addresses.Select(a => a.PrimaryState),
                     z.SubOrder.Cost,        
                     z.SubOrder.SubOrderId,
                     z.Sequence});   
于 2013-02-11T19:15:27.977 回答
0

Expand查询上完成。请参阅如何:加载相关实体(WCF 数据服务)

你想要这样的东西:

Orders
  .Expand("StateRef")
  .Select(z => new { ... } );
于 2013-02-08T22:39:47.177 回答