我设置了LDAP DirContext.search(...)以忽略引用,
但是当我调用NamingEnumeration.hasMore()时,仍然收到引用异常。

Exception in thread "main" javax.naming.PartialResultException: Unprocessed Continuation Reference(s); remaining name 'DC=company,DC=com'
    at com.sun.jndi.ldap.LdapCtx.processReturnCode(LdapCtx.java:2846)
    at com.sun.jndi.ldap.LdapCtx.processReturnCode(LdapCtx.java:2820)
    at com.sun.jndi.ldap.LdapNamingEnumeration.getNextBatch(LdapNamingEnumeration.java:129)
    at com.sun.jndi.ldap.LdapNamingEnumeration.hasMoreImpl(LdapNamingEnumeration.java:198)
    at com.sun.jndi.ldap.LdapNamingEnumeration.hasMore(LdapNamingEnumeration.java:171)

我可以告诉DirContext.search忽略引荐,
这样NamingEnumeration.hasMore()返回false而不是抛出异常?

这是片段:
import javax.naming.*;
import javax.naming.directory.*;

Properties p = new Properties();
p.setProperty(Context.INITIAL_CONTEXT_FACTORY, ldapInitContextFactory);
p.setProperty(Context.PROVIDER_URL, ldapURL);
p.setProperty(Context.SECURITY_CREDENTIALS, ldapPassword);
p.setProperty(Context.SECURITY_PRINCIPAL, ldapUser);
p.setProperty(Context.SECURITY_AUTHENTICATION, "simple");
p.setProperty(Context.REFERRAL, "ignore");
DirContext ctx = new InitialDirContext(p);

SearchControls searchControls = new SearchControls();
searchControls.setSearchScope(SearchControls.SUBTREE_SCOPE);
searchControls.setDerefLinkFlag(false);

NamingEnumeration e = ctx.search(ldapBaseDN, ldapQuery, null, searchControls);

for (; e.hasMore();) {
    SearchResult sr = (SearchResult) e.next();
    System.out.println("\nSearch Result: " + sr.getName());
}

注意:如果启用引用,则在调用NamingEnumeration.hasMore()时会收到LdapReferralException异常。

最佳答案

javax/naming/NamingEnumeration.java
public interface NamingEnumeration<T> extends Enumeration<T> {
    public boolean hasMore() throws NamingException;
    public T next() throws NamingException;
    ...
}

java/util/Enumeration.java
public interface Enumeration<E> {
    boolean hasMoreElements();
    E nextElement();
}

调用e.hasMoreElements()而不是e.hasMore()解决了此问题。
也就是说,当存在引用时,它返回false(而不是引发异常)。

但是,从我们可能会丢失其他实际NamingException异常(例如CommunicationException)的意义上来说,这不是最佳解决方案。

我仍然想找到告诉DirContext.search忽略引荐的正确方法,
因此NamingEnumeration.hasMore()返回false而不是引发异常。
任何想法?

我从http://download.java.net/openjdk/jdk6/-> openjdk-6-src-b24-14_nov_2011.tar.gz下载了JDK源代码。
但是,此源代码与JDK二进制文件不完全对应,
how to find the exact sources of a JDK1.6 binary (including com.sun.jndi.*)

从下面的此JDK源代码看来,当存在引用时,似乎不可能获得“假”而不是异常。
./jdk/src/share/classes/com/sun/jndi/ldap/LdapCtx.java
protected void processReturnCode(LdapResult res, Name resolvedName, Object resolvedObj, Name remainName, Hashtable envprops, String fullDN) throws NamingException {
    NamingException e;

    switch (res.status) {
    case LdapClient.LDAP_SUCCESS:

        // handle Search continuation references
        if (res.referrals != null) {
            msg = "Unprocessed Continuation Reference(s)";

            if (handleReferrals == LdapClient.LDAP_REF_IGNORE) {
                e = new PartialResultException(msg);
                break;
            }
        [...]
    }
    [...]
    throw e;
}

但我仍然不确定。

我认为我们可以告诉DirContext.search忽略引荐是有道理的,
因此NamingEnumeration.hasMore()返回false而不是引发异常。

任何想法?

07-25 22:22