我可以从 DirectorySearcher 获得超过 1000 条记录吗?

Posted

技术标签:

【中文标题】我可以从 DirectorySearcher 获得超过 1000 条记录吗?【英文标题】:Can I get more than 1000 records from a DirectorySearcher? 【发布时间】:2010-09-10 13:52:13 【问题描述】:

我刚刚注意到结果的返回列表限制为 1000。我的域(HUGE 域)中有超过 1000 个组。如何获得超过 1000 条记录?我可以从以后的记录开始吗?我可以把它分成多个搜索吗?

这是我的查询:

DirectoryEntry dirEnt = new DirectoryEntry("LDAP://dhuba1kwtn004");
string[] loadProps = new string[]  "cn", "samaccountname", "name", "distinguishedname" ;
DirectorySearcher srch = new DirectorySearcher(dirEnt, "(objectClass=Group)", loadProps);
var results = srch.FindAll();

我尝试设置 srch.SizeLimit = 2000;,但这似乎不起作用。有什么想法吗?

【问题讨论】:

【参考方案1】:

您需要将 DirectorySearcher.PageSize 设置为非零值才能获得所有结果。

顺便说一句,您还应该在使用完 DirectorySearcher 后将其丢弃

using(var srch = new DirectorySearcher(dirEnt, "(objectClass=Group)", loadProps))

    srch.PageSize = 1000;
    var results = srch.FindAll();

API 文档不是很清楚,但本质上是:

当您进行分页搜索时,SizeLimit 将被忽略,并在您遍历 FindAll 返回的结果时返回所有匹配的结果。结果将从服务器一次检索一个页面。我在上面选择了 1000 的值,但如果愿意,您可以使用较小的值。权衡是:使用较小的 PageSize 会更快地返回每页结果,但在迭代大量结果时需要更频繁地调用服务器。

默认情况下不分页搜索 (PageSize = 0)。在这种情况下,最多返回 SizeLimit 结果。

正如 Biri 指出的,处理 FindAll 返回的 SearchResultCollection 很重要,否则您可能会出现内存泄漏as described in the Remarks section of the MSDN documentation for DirectorySearcher.FindAll。

在 .NET 2.0 或更高版本中帮助避免这种情况的一种方法是编写一个自动处理 SearchResultCollection 的包装器方法。这可能类似于以下内容(或者可能是 .NET 3.5 中的扩展方法):

public IEnumerable<SearchResult> SafeFindAll(DirectorySearcher searcher)

    using(SearchResultCollection results = searcher.FindAll())
    
        foreach (SearchResult result in results)
        
            yield return result;        
         
     // SearchResultCollection will be disposed here

然后您可以按如下方式使用它:

using(var srch = new DirectorySearcher(dirEnt, "(objectClass=Group)", loadProps))

    srch.PageSize = 1000;
    var results = SafeFindAll(srch);

【讨论】:

如果foreach提前终止怎么办?结果集合还会发布吗? @kerem 也许这不是对您的问题的及时回应,但是即使出现异常,using 也会导致调用Dispose()(使用finally 块[@987654322 @这样做)) @Joe 出于某种原因,我发现“默认情况下不分页搜索(PageSize = 0)。在这种情况下,最多返回 SizeLimit 结果。”使用具有底层DirectorySearcherPrincipalSearcher 时似乎不正确。它只是继续枚举结果。也许服务器端或PrincipalSearcher 会以某种方式覆盖它...... @GuacoIV - PrincipalSearcher 很可能会覆盖默认行为,特别是因为它没有公开 PageSizeSizeLimit 属性。您可以通过检查PrincipalSearcher.GetUnderlyingSearcher 方法返回的DirectorySearcher 来确认这一点,看看它是如何配置的。 你会在网上找到很多关于这个的帖子。有些确实令人困惑,似乎暗示无论您做什么,服务器端设置都会破坏您的一天。从 AD 2012 开始,这是使用 DirectorySearcher 检索任意数量对象的正确答案。我已经尝试过 18k 的结果。 srch.PageSize = int.MaxValue。工作一种享受。【参考方案2】:

我尝试使用给定的解决方案来实现分页,但它不起作用。 我设置 pageSize = 100;我通过 searchResult 获取所有项目。

dirSearcher = new DirectorySearcher(目录); dirSearcher.Filter = ("(|(objectClass=volume). (objectClass=user)(objectClass=printQueue)(objectClass=computer). (objectClass=organizationalUnit)(objectClass=Group))");

            dirSearcher.PageSize = 100;
            dirSearcher.SearchScope = System.DirectoryServices.SearchScope.Subtree;
            dirSearcher.ServerTimeLimit = new TimeSpan(1000);

            //dirSearcher.VirtualListView = new DirectoryVirtualListView(0, 100, 1);


            using (SearchResultCollection results = dirSearcher.FindAll())
            
                foreach (SearchResult result in results)
                
                    DirectoryEntry ent = result.GetDirectoryEntry();
                    ADItem ProviderItem = Context.ConvertToItem(ent, true);

                    if (ProviderItem != null)
                    
                        ProviderItem.IsPartialData = true;
                        ProviderItems.Add(ProviderItem);
                    
                

            
        

【讨论】:

以上是关于我可以从 DirectorySearcher 获得超过 1000 条记录吗?的主要内容,如果未能解决你的问题,请参考以下文章

DirectorySearcher.Filter 属性(转)

Actiove Directory的DirectoryEntry与DirectorySearcher初识及Filter语法

实现域登录 --- 介绍

活动目录 - 获取密码即将到期的用户列表

LdapConnection 与 DirectoryEntry

我可以从我已经在 mapView 上获得的方向获得大致的旅行时间吗?