简体   繁体   English

C#LINQ嵌套选择查询

[英]C# LINQ nested select query

I have a scenario in following nested 我有一个跟随嵌套的场景

--Orders (List) 
 ----Products (List)
 ------Manufacturers (List) 
       FIELDS 
        -Name
        -Address
        -City 

In this scenario, I would need to execute query which will filter on City of Manufacturers and returns Orders, Products & only matching city manufacturers 在这种情况下,我需要执行查询,该查询将过滤制造 商城市并返回Orders, Products & only matching city manufacturers

I tried to put following query, however I am getting all list of Products even though city doesn't match to Manufacturers . 我尝试进行以下查询,但即使城市与制造商不匹配,我也会获得所有产品列表。

var filteredOrders = from o in Orders
                    from t in o.Products                           
                    where t.Manufacturers.Any(v => v.City == "Hartford")
                    select o;

Or even if I change from select o to 'select t.Manufacturers' I am getting all list of Manufacturers irrespective of city filter. 或者即使我从select o更改为'select t.Manufacturers',我也会获得所有制造商列表,而不管城市过滤器。

Luckily I got W3school SQL sample which matches to my scenario. 幸运的是,我得到了与我的场景相匹配的W3school SQL样本。 https://www.w3schools.com/sql/trysql.asp?filename=trysql_op_or https://www.w3schools.com/sql/trysql.asp?filename=trysql_op_or

SQL Query: SQL查询:

SELECT o.OrderId, p.ProductName, s.* 
FROM [Orders] o 
JOIN OrderDetails od ON o.OrderId = od.OrderId AND o.orderId = 10248 
JOIN Products p ON od.ProductId = p.ProductId
JOIN Suppliers s ON p.SupplierId = s.SupplierId and s.City ='Singapore'    

I would flatten everything and then only filter on cities you want: 我会压扁一切,然后只过滤你想要的城市:

class Manufacturer
{
    public string Name;
    public string Address;
    public string City;
}

class Product
{
    public Manufacturer[] Manufacturers;
}

class Order
{
    public Product[] Products;
}

static void Main(string[] args)
{
    var cities = new string[] { "a", "b" };
    Order[] orders = null;
    orders.SelectMany(o => o.Products.SelectMany(p => p.Manufacturers.Select(m => new { o, p, m })))
        .Where(g => cities.Contains(g.m.City))
        .ToList();
    }

Alternatively, if you want to return new Order s (because they have a different Products , it MUST point to a newly allocated Object ) you could have this instead: 或者,如果您想要返回 Order (因为它们具有不同的Products ,它必须指向新分配的Object ),您可以改为:

var newOrders = orders.Select(o => new Order()
{
    Products = o.Products
    .Select(p => new Product()
    {
        Manufacturers = p.Manufacturers.Where(m => cities.Contains(m.City)).ToArray()
    })
    .Where(m => m.Manufacturers.Length > 0).ToArray()
}).Where(p => p.Products.Length > 0).ToArray();

You are applying your City filter wrong. 您正在应用错误的城市过滤器。 It is this line. 就是这条线。

where t.Manufacturers.Any(v => v.City == "Hartford")

Any return true , at least one of the manufacturers has City property as "Hartford" so basically your query is something like this Any回报true ,至少有一家制造商将City的财产称为“Hartford”,所以基本上你的查询是这样的

var filteredOrders = from o in Orders
                from t in o.Products                           
                where true//←This is the problem
                select o;

What you need to do is in fact 你需要做的事实上是

where t.Manufacturers.City == "Hartford"

I hope this helps 我希望这有帮助

Example: 例:

var cityNames = new List<string> {"New York",
                                  "Atlanta",
                                  "Hartford",
                                  "Chicago"
                                  };
var anyResult = cityNames.Any(x=>x== "Hartford"); //TRUE
var whereResult = cityNames.Where(x => x == "Hartford"); //IEnumerable<string>, in this case only one element

I cannot think of a way which can completely avoid creating new objects, as the parent object's list property cannot be filtered directly. 我无法想到一种可以完全避免创建新对象的方法,因为父对象的list属性不能直接过滤。 You can make use of the same class though. 你可以使用同一个类。

Also I use two separate queries in order to create a new list in parent / grandparent object. 此外,我使用两个单独的查询,以便在父/祖父对象中创建一个新列表。

I have made a small demo to demonstrate the idea (below has equivalent code): http://ideone.com/MO6M6t 我做了一个小演示来演示这个想法(下面有相同的代码): http//ideone.com/MO6M6t

The city I try to select is "tmp" which only under parent p3 , which only belongs to grand parent g1 , g3 我试图选择的城市是"tmp" ,它只在父p3 ,它只属于g1g3

The expected output is: 预期的产出是:

g1
    p3
        tmp

g3
    p3
        tmp
using System;
using System.Collections.Generic;
using System.Linq;

public class Test
{
    public class GrandParent{
        public List<Parent> parentList{ get; set; }
        public string name{ get; set; }
        public GrandParent(string name){
            this.name = name;
            this.parentList = new List<Parent>();
        }
    }
    public class Parent{
        public List<Child> childList{ get; set;}
        public string name{ get; set; }
        public Parent(string name){
            this.name = name;
            this.childList = new List<Child>();
        }
    }
    public class Child{
        public string city{ get; set;}
        public Child(string city){
            this.city = city;
        }
    }
    public static void Main()
    {
        Child c1 = new Child("ABC"), c2 = new Child("123"), c3 = new Child("tmp");
        Parent p1 = new Parent("p1"), p2 = new Parent("p2"), p3 = new Parent("p3");
        GrandParent g1 = new GrandParent("g1"), g2 = new GrandParent("g2"), g3 = new GrandParent("g3");

        p1.childList.Add(c1); p1.childList.Add(c2); 
        p2.childList.Add(c2); 
        p3.childList.Add(c3);

        g1.parentList.Add(p1); g1.parentList.Add(p2); g1.parentList.Add(p3);
        g2.parentList.Add(p2);
        g3.parentList.Add(p3);

        List<GrandParent> repo = new List<GrandParent>{g1, g2, g3};

        var filteredParents = from g in repo
                              from p in g.parentList
                              where p.childList.Any(c => c.city == "tmp")
                              select new Parent(p.name){
                                 childList = p.childList.Where(c => c.city == "tmp").ToList()
                              };

        var filteredGrandParents = from g in repo
                                   from p in g.parentList
                                   where filteredParents.Any(fp => fp.name == p.name)
                                   select new GrandParent(g.name){
                                       parentList = g.parentList.Where(pp => filteredParents.Any(fp => fp.name == pp.name)).ToList()
                                   };

        foreach(var g in filteredGrandParents){
            Console.WriteLine(g.name);
            foreach(var p in g.parentList){
                Console.WriteLine("\t" + p.name);
                foreach(var c in p.childList){
                    Console.WriteLine("\t\t" + c.city);
                }
            }
            Console.WriteLine();
        }
    }
}

I finally tried to put everything together and got the expected output. 我终于试着把所有东西放在一起,得到了预期的输出。

var fp = orders.Select(o =>
            {
                o.products = o.products.Select(p =>
                {
                    p.manufacturers.RemoveAll(m => m.City != "Hartford");
                    return p;
                }).ToList();

                return o;
            });

Please suggest if anyone has better solution 请建议是否有人有更好的解决方案

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM