web-dev-qa-db-ja.com

DirectorySearcherから1000を超えるレコードを取得できますか?

結果の戻りリストが1000に制限されていることに気付きました。ドメイン(巨大なドメイン)に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;を設定しようとしましたが、うまくいかないようです。何か案は?

68
naspinski

すべての結果を取得するには、DirectorySearcher.PageSizeをゼロ以外の値に設定する必要があります。

ところで、それが終わったら、DirectorySearcherも破棄する必要があります

using(var srch = new DirectorySearcher(dirEnt, "(objectClass=Group)", loadProps))
{
    srch.PageSize = 1000;
    var results = srch.FindAll();
}

APIドキュメントはあまり明確ではありませんが、基本的には:

  • ページ検索を実行すると、SizeLimitは無視され、FindAllによって返された結果を反復処理すると、一致するすべての結果が返されます。結果は、一度に1ページずつサーバーから取得されます。上記の値1000を選択しましたが、必要に応じてより小さい値を使用できます。トレードオフは次のとおりです。小さなPageSizeを使用すると、結果の各ページがより速く返されますが、多数の結果を反復処理する場合、サーバーへの呼び出しがより頻繁に必要になります。

  • デフォルトでは、検索はページングされません(PageSize = 0)。この場合、SizeLimitまでの結果が返されます。

Biriが指摘したように、FindAllによって返されたSearchResultCollectionを破棄することが重要です。そうしないと、メモリリークが発生する可能性があります DirectorySearcher.FindAllのMSDNドキュメントの備考セクションで説明

.NET 2.0以降でこれを回避する1つの方法は、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);
}
172
Joe