c# - Linq TotalCount 为 "Grandchildren",具有空检查和属性筛选器

标签 c# linq c#-4.0 linq-to-objects

提供以下类(class):

[DebuggerDisplay("CustomerKey = {CustomerKey}")]
public class Customer
{
    public Customer()
    {
        this.Orders = new List<Order>();
    }
    public int CustomerKey { get; set; }
    public ICollection<Order> Orders { get; set; }
}

[DebuggerDisplay("OrderKey = {OrderKey}, OrderDateOffset='{OrderDateOffset}'")]
public class Order
{
    public Order()
    {
        this.OrderDetails = new List<OrderDetail>();
    }
    public int OrderKey { get; set; }
    public Customer ParentCustomer { get; set; }
    public DateTimeOffset OrderDateOffset { get; set; }
    public ICollection<OrderDetail> OrderDetails { get; set; }
}

[DebuggerDisplay("OrderDetailKey='{OrderDetailKey}', ProductKey='{ProductKey}', Quantity='{Quantity}', UnitPrice='{UnitPrice}, BackOrdered='{BackOrdered}'")]
public class OrderDetail
{
    public int OrderDetailKey { get; set; }
    public Order ParentOrder { get; set; }
    public int ProductKey { get; set; }
    public int Quantity { get; set; }
    public Decimal UnitPrice { get; set; }
    public bool BackOrdered { get; set; }
}

考虑到以下水分:

            ICollection<Customer> customers = new List<Customer>();

            Customer customerOne = new Customer() { CustomerKey = 11111 };
            /**/
            Order orderOne = null;
            customerOne.Orders.Add(orderOne);/* note, the item added to the collection is null) */
            /**/
            Order orderTwo = new Order() { OrderKey = 22222, OrderDateOffset = DateTimeOffset.Now };
            orderTwo.OrderDetails = null;/* Note, the child collection is null */
            customerOne.Orders.Add(orderTwo);
            /**/
            Order orderThree = new Order() { OrderKey = 22223, OrderDateOffset = DateTimeOffset.Now };
            orderThree.OrderDetails.Add(null); /* note, the item added to the collection is null) */
            customerOne.Orders.Add(orderThree);
            /**/
            Order orderFour = new Order() { OrderKey = 22221, OrderDateOffset = DateTimeOffset.Now };
            orderFour.OrderDetails.Add(new OrderDetail() { OrderDetailKey = 33333, ProductKey = 11, Quantity = 1, UnitPrice = 1M, BackOrdered = false });
            orderFour.OrderDetails.Add(new OrderDetail() { OrderDetailKey = 33334, ProductKey = 12, Quantity = 2, UnitPrice = 2M, BackOrdered = false });
            orderFour.OrderDetails.Add(new OrderDetail() { OrderDetailKey = 33335, ProductKey = 13, Quantity = 3, UnitPrice = 3M, BackOrdered = true });
            customerOne.Orders.Add(orderFour);

            customers.Add(customerOne);
            customers.Add(null);/* note, the item added to the collection is null) */

我正在尝试获取所有延期交货的订单详细信息的总数。

我的预 linq 代码:

            int totalBackOrderedCount = 0;
            if (null != customers)
            {
                foreach (Customer cust in customers)
                {
                    if (null != cust)
                    {
                        if (null != cust.Orders)
                        {
                            foreach (Order ord in cust.Orders)
                            {
                                if (null != ord)
                                {
                                    if (null != ord.OrderDetails)
                                    {
                                        foreach (OrderDetail ordDet in ord.OrderDetails)
                                        {
                                            if (null != ordDet)
                                            {
                                                if (ordDet.BackOrdered)
                                                {
                                                    totalBackOrderedCount++;
                                                }
                                            }
                                            else
                                            {
                                                Console.WriteLine("ordDet was null, good thing I coded for it");
                                            }
                                        }
                                    }
                                    else
                                    {
                                        Console.WriteLine("ord.OrderDetails was null, good thing I coded for it");
                                    }
                                }
                                else
                                {
                                    Console.WriteLine("ord was null, good thing I coded for it");
                                }
                            }
                        }
                    }
                    else
                    {
                        Console.WriteLine("cust was null, good thing I coded for it");
                    }
                }
            }

            Console.WriteLine("totalBackOrderedCount={0}", totalBackOrderedCount);

我对 linq 替代方案的尝试:

            int linqyTotalBackOrderedCountOne =
               (
                 from cust in customers
                 from ord in (cust.Orders ?? Enumerable.Empty<Order>()).DefaultIfEmpty()
                 from ordDet in (ord.OrderDetails.Where(od => od.BackOrdered == true) ?? Enumerable.Empty<OrderDetail>()).DefaultIfEmpty()
                 where cust != null && (null != ord) && (null != ordDet)
                 select ordDet
               ).Count();

            int linqyTotalBackOrderedCountTwo =
                   (
                     from cust in customers
                     from ord in (cust.Orders ?? Enumerable.Empty<Order>()).DefaultIfEmpty()
                     from ordDet in (ord.OrderDetails.Where(od => od.BackOrdered == true) ?? Enumerable.Empty<OrderDetail>()).DefaultIfEmpty()
                     where cust != null && (null!=cust.Orders) && (null!=ord) && (null!=ord.OrderDetails) && (null!=ordDet)
                     select ordDet
                   ).Count();

linq 中是否有一种“空友好”的方式来获取 child 的 child ......?

最佳答案

您应该能够执行类似的操作,而无需担心空值。

var count = customers.Where (c => c.Orders != null)
    .SelectMany (c => c.Orders.Where(o => o.OrderDetails != null)
        .SelectMany (o => o.OrderDetails))
    .Count (c => c.BackOrdered);

上面的内容很接近......这是最终答案:

            int itWorksCount = customers.Where(c => null != c && null != c.Orders)
                    .SelectMany(c => c.Orders.Where(o => null != o && null != o.OrderDetails)
                        .SelectMany(o => o.OrderDetails.Where(ordDet => null != ordDet)))
                    .Count(c => c.BackOrdered);

关于c# - Linq TotalCount 为 "Grandchildren",具有空检查和属性筛选器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29929578/

相关文章:

c# - 在 List<T> 中查找项目

c# - ConfigureAwait(false) 维护线程身份验证,但默认情况下它不会

wpf - 列表框项目源不随源更新

.net - RNGCryptoServiceProvider 使用哪种伪随机数生成算法?

c#反序列化对象错误的JSON列表

c# - 如何在动态选择中使用表达式创建 "inline if statement"以进行空检查

C# Generic ToString Join if Array or IEnumerable

c# - 无法访问 Newtonsoft.Json.Linq.JProperty 上的子值 - 使用 LinQ 检查 JObject 时发生错误

c# - 如何编辑所有特殊 html 标签的属性(如 a)

c# - 如何检查 net.tcp WCF 服务的可用性