如何在一个选择(Ef核心)中联接多个表?

时间:2020-06-11 09:54:21

标签: c# asp.net-core asp.net-web-api entity-framework-core

我有一个数据库模型: enter image description here

表“产品”和“属性”无关。但是每种产品都可以有一个属性列表。

为了向客户发送带有附加属性和值的产品列表,我实现了以下逻辑: API控制器方法:

[HttpGet("get")]
[AllowAnonymous]
public async Task<IActionResult> Get()
{
  var productList = await _db.Product
    .Include(p => p.Category)
    .Include(p => p.ProductType)
    .Take(25)
    .ToListAsync();

  var productListJson = await HandleProductEntities(productList);

  return Ok(productListJson.ToString());
}

以及形成具有属性的产品json-object的方法:

private async Task<JArray> HandleProductEntities(IEnumerable<Product> productList)
{
  if (productList == null) throw new ArgumentNullException(nameof(productList));
  var productListJson = new JArray();
  foreach (var productEntity in productList)
  {
    var attributesList = await _db.ProductAttributeValues
      .Where(pav => pav.ProductId == productEntity.Id)
      .Select(pav => new
      {
        Id = pav.AttributeValue.Attribute.Id,
        Name = pav.AttributeValue.Attribute.Name,
        DisplayName = pav.AttributeValue.Attribute.DisplayName,
        Value = pav.AttributeValue.Value,
        ValueDescription = pav.AttributeValue.Description
      }).ToListAsync();

    var jsonSerializerSettings = new JsonSerializerSettings
    {
      ContractResolver = new CamelCasePropertyNamesContractResolver()
    };
    var productJson = JObject.Parse(JsonConvert.SerializeObject(productEntity, jsonSerializerSettings));

    var attrArrayJson = JArray.Parse(JsonConvert.SerializeObject(attributesList, jsonSerializerSettings));
    foreach (var token in attrArrayJson)
    {
      var paramName = (string) token["name"];
      if (paramName == null)
      {
        throw new NullReferenceException();
      }

      productJson.Add(paramName, token);
    }

    productListJson.Add(productJson);
  }

  return productListJson;
}

但是此逻辑过载。使用EF核心,可能有更简单的方法来从数据库中获取具有属性的产品。有人可以告诉我该怎么做吗?

我试图这样做:

var resultTest = from productAttributeValue in _db.ProductAttributeValues
  join attributeValue in _db.AttributeValue on productAttributeValue.AttributeValueId equals attributeValue.Id
  join product in _db.Product on productAttributeValue.ProductId equals product.Id
  join attribute in _db.Attribute on attributeValue.AttributeId equals attribute.Id
  select new
  {
    Name = product.Name,
    Attribute = attribute.Name,
    AttributeValue = attributeValue,
  };

但是无法对每种产品的属性进行分组。

1 个答案:

答案 0 :(得分:0)

from product in _db.Product
join pav in _db.ProductAttributeValues on product.Id equals 
productAttributeValue.ProductId into productAttributeValueList
from productAttributeValue in productAttributeValueList.DefaultIfEmpty()
join attributeValue in _db.AttributeValue on productAttributeValue.AttributeValueId equals attributeValue.Id
join attribute in _db.Attribute on attributeValue.AttributeId equals attribute.Id
select new 
{
    Name = product.Name,
    Attribute = attribute.Name,
    AttributeValue = attributeValue
}