繁体   English   中英

根据条件将对象列表过滤到另一个列表中的更快方法

[英]Faster way to filter a list of objects in to another list based on a condition

我有一个列表(ios联系人),我需要根据匹配字符串开头的名字,姓氏和电子邮件进行过滤。 我有大约5000个联系人,目前大约需要2秒钟来过滤结果。 这是我的代码。

    var personList = people.FindAll (p => 
                        (p.LastName != null && p.LastName.IndexOf (findstr, StringComparison.OrdinalIgnoreCase) == 0) || (p.FirstName != null && p.FirstName.IndexOf (findstr, StringComparison.OrdinalIgnoreCase) == 0
                            || (p.GetEmails ().ToList ().Count > 0 && (p.GetEmails ().ToList ().FindAll (e => e.Value.IndexOf (findstr, StringComparison.OrdinalIgnoreCase) == 0).Count > 0)))

                    );

谁能建议一个更快的方法来做到这一点? 谢谢

代替FindAll ,先调用ToList然后再调用Count ,请使用:

var personList = people.Where(p => (p.LastName != null 
    &&  p.LastName.IndexOf(findstr, StringComparison.OrdinalIgnoreCase) == 0)
    || (
        p.FirstName != null && p.FirstName.IndexOf(findstr, StringComparison.OrdinalIgnoreCase) == 0
        || (p.GetEmails().Count > 0 && (p.GetEmails()
                                            .Where(e => 
                                                e.Value.IndexOf(findstr, StringComparison.OrdinalIgnoreCase) == 0).Count > 0))
             ));

使用IndexOf(..) == 0进行比较与询问该字符串(名称,姓氏或电子邮件)是否以给定的findstr开头相同。 相反,如果您想知道字符串是否包含findstr使用String.ContainsIndexOf(..) != -1

为了简化代码,我还将这些谓词分开:

Func<string, bool> filter = s => !String.IsNullOrWhiteSpace(s) && s.StartsWith(findstr, StringComparison.OrdinalIgnoreCase);

var personList = people.Where(p => filter(p.FistName) || filter(p.LastName) || p.GetEmails().Select(e => e.Value).Any(filter));

现在,如果您愿意,可以并行进行:

var personList = people.AsParallel(). Where(p => filter(p.FistName) || filter(p.LastName) || p.GetEmails().Select(e => e.Value).Any(filter));

您可以使用:

  • Any()代替ToList().Count > 0
  • First() ,然后检查它是否符合条件而不是IndexOf (findstr, StringComparison.OrdinalIgnoreCase) == 0

您可以在此处找到队列的完整列表。

暂无
暂无

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

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