2

我想编写一个优雅的 linq 查询来处理以下 SAMPLE 对象模型:

    class Category
    {
        public string Name { get; set; }        
        public IList<Product> Products { get; set;}        
    }

    class Product
    {
        public string Title { get; set; }
        public IList<Photo> Photos { get; set; }
    }

    class Photo
    {
        public int Id { get; set; }
    }

我构建了以下查询来获取照片 ID:

    var query = from category in Factory.GetCategories()
                where category.Name == "Cameras"
                select (from product in category.Products
                        where product.Title == "Sony"
                        select (from photo in product.Photos
                                select photo.Id)
                               );

    var v = query.ToList();

目前查询没有正确投影,我必须向每个子选择添加一个 FirstOrDefault() !:

var query = from category in Factory.GetCategories()
                where category.Name == "Cameras"
                select (from product in category.Products
                        where product.Title == "Sony"
                        select (from photo in product.Photos
                                select photo.Id).FirstOrDefault()
                               ).FirstOrDefault();

var v = query.ToList();

有一个更好的方法吗?忽略我们不处理数据库并且PK/FK 不在游戏中的事实。

当我可以在 linq 查询中做同样的事情时,我真的想避免编写一个大的 for 循环:

foreach (var category in Factory.GetCategories())
            {
                if (category.Name == "Camera")
                {
                    foreach (var product in category.Products)
                    {
                        if (product.Title == "Sony")
                        {
                            foreach (var photo in product.Photos)
                            {
                                //get data
                                int id = photo.Id;
                            }
                        }
                    }
                }                
            }

实际的实现比这个简单的对象模型更复杂。我想从这个简单的示例中获得基本思想,因此我将其应用于我的真实对象模型。

干杯!

4

4 回答 4

5

你只想要扁平的ID吗?

        var query = from category in Factory.GetCategories()
                    where category.Name == "Cameras"
                    from product in category.Products
                    where product.Title == "Sony"
                    from photo in product.Photos
                    select photo.Id;
于 2009-02-25T15:12:35.580 回答
2

嗯,是的 - 你必须添加一个 FirstOrDefault 因为没有像“那个”照片 ID 这样的东西。每个相机可能有很多照片 - 你想要哪张?

如果您只关心第一张照片,那很好 - 尽管多个“来自”子句会让您的生活更轻松:

var query = from category in Factory.GetCategories()
            where category.Name == "Cameras"
            from product in category.Products
            where product.Title == "Sony"
            select product.Photos.Select(photo => photo.Id)
                                 .FirstOrDefault();

请注意,对于没有照片的产品,这将返回 0。那是你要的吗?

如果您能更清楚地了解您的要求,我们将更有能力为您提供帮助。

编辑:如果您只想要任何索尼相机的第一个 ID,请使用:

var query = from category in Factory.GetCategories()
            where category.Name == "Cameras"
            from product in category.Products
            where product.Title == "Sony"
            from photo in product.Photos
            select photo.Id;
var firstId = query.FirstOrDefault();
于 2009-02-25T15:13:19.993 回答
1

与其他所有人的答案相同,但使用扩展方法调用而不是查询理解。 SelectMany允许您“解包”子集合并在该级别继续查询。

var query = Factory.GetCategories()
  .Where(category => category.Name == "Cameras")
  .SelectMany(category => category.Products)
  .Where(product => product.Title == "Sony")
  .SelectMany(product => product.Photos)
  .Select(photo => photo.Id);
于 2009-02-25T16:14:08.690 回答
0
        var ids = from category in Factory.Categories
                  where category.Name == "Cameras"
                  from product in category.Products
                  where product.Title == "Sony"
                  from photo in product.Photos
                  select photo.Id;

返回所有照片的id,没有照片时可枚举为空。如果你想要第一个,或者默认为 0 :

var result = ids.FirstOrDefault();
于 2009-02-25T15:20:24.077 回答