5

我用 Durandal/breeze 开发了一个 asp.net 解决方案。

这是我获取所有托运人的代码:

var query = EntityQuery.from('Shippers')
               .select('id, name, street, city');

return manager.executeQuery(query)
        .then(querySucceeded)
        .fail(queryFailed);

以下是相关模型:

public class Shipper
{
    [Key]
    public int Id { get; set; }
    public string Name { get; set; }
    public string Street { get; set; }
    public string Number { get; set; }
    public City City { get; set; }
}

public class City
{
    public int Id { get; set; }        
    public string Name { get; set; }
    public string PostCode { get; set; }
    public Country Country { get; set; }
}

现在我还需要包括国家

public class Country
{
    [Key]
    public int Id { get; set; }
    public string Code { get; set; }
    public string Name { get; set; }
}

但是通过实际查询,我没有得到国家。

我尝试:

var query = EntityQuery.from('Shippers')
               .select('id, name, street, city')
               .expand('City.Country');

但我得到了错误:

use of both 'expand' and 'select' in the same query is not currently supported

我的问题:如何获得国家?


更新

正如 Jay 建议的那样,我们可以这样做:

var query = EntityQuery.from('Shippers')
       .select('id, name, street, city, city.country')

现在,我得到了一个city_Country对象:

在此处输入图像描述

我不明白为什么我们得到这个city_Country,因为城市\国家对象中已经有国家数据:

在此处输入图像描述

此外,它给我带来了问题,因为我的下一条语句尝试将我的 dto 映射到我的实体中,并且该city_Country对象在我的实体中不存在,并且在映射时发生错误。

下面我们看到我的实体对象并且没有city_Country对象:

在此处输入图像描述

我必须在我的映射上做一些特别的事情来避免它吗?

以下是我的映射操作功能:

function mapToEntity(entity, dto) {
     // entity is an object with observables
     // dto is from json
     for (var prop in dto) {
          if (dto.hasOwnProperty(prop)) {
             entity[prop](dto[prop]);
          }
     }
     return entity;
}
4

2 回答 2

8

我不确定使用投影来“部分”填充实体是一种好习惯,这就是你正在做的事情。Breeze 将自动映射它在您的结果集中找到的任何真正的“实体”。那么为什么不简单地使用

var query = EntityQuery.from('Shippers')
    .where(...)
    .expand('city.country');

直接结果将是托运人的集合,但每个托运人的城市和嵌套国家属性也将完全解析为实体。它们将通过返回的托运人的导航提供,但如果您想直接查询它们,它们也将在 entityManager 缓存中可用。

第二个注意事项:Breeze“扩展”语义具有与实体框架相同的限制。这意味着我们不能扩展投影的属性。

所以你可以跳过投影(如上)

var query = EntityQuery.from('Shippers')
   .where(...)
   .expand('city.country')

并获得完整的“托运人”实体,城市上的“城市”和“国家”属性都被填充。......或者你可以做一个投影,在这种情况下你自己执行扩展的等价物。即

在这种情况下,您将

var query = EntityQuery.from('Shippers')
  .select('id, name, street, city, city.country')

在这种情况下,结果集中的每个项目都将包含 5 个属性。请注意,在这种情况下,只有“city”和“city.country”属性将被添加到 entityManager 的缓存中,因为它们是结果集中唯一的“真实”实体。即没有托运人的

要明确的想法是查询的“结果”和查询的“副作用”是不同的。查询的顶级结果将完全符合您的预期。查询的“副作用”是您执行的任何“扩展”的结果。这些不会改变查询的形状,它们只是改变结果中任何嵌套“实体”属性的分辨率。

希望这可以帮助。

于 2013-04-04T17:49:51.297 回答
1

我们确实允许这样做:

var query = EntityQuery.from('Shippers')
           .select('id, name, street, city, city.country')
于 2013-04-03T17:28:34.157 回答