JNDI全攻略之(二)
關鍵字:JNDI,J2EE,Java,命名和目錄接口,Java Naming and Directory Interface
摘要:本文詳細介紹了JNDI的目錄相關內容,并以DNS Service Provider為例進行了示例代碼的演示.本文為系列文章的第二篇,JNDI的基礎內容請見本系列的第一篇
總述:
目錄(Directory)可看作是對命名(Naming)的一個擴充,一個目錄對象不僅像命名一樣,而且還提供的對屬性(Attributes)的操作.由API文檔可知,javax.naming.directory.DirContext 類擴展自Context接口,同樣,javax.naming.directory.InitialDirContext也擴展自javax.naming.InitialContext,由此也可看出目錄操作完全支持命名操作。下面給出一個DNS Service Provider例子以演示有關目錄的一些操作:
package com.sily.jndi;
import java.util.Properties;
/**
* Description:
*
* @author shizy
* @version 1.0 date:2005-11-17
*/
public class TestDNSJndi {
public static void main(String[] args) throws Exception {
Properties env = new Properties();
env.put(Context.INITIAL_CONTEXT_FACTORY,
"com.sun.jndi.dns.DnsContextFactory");
//此IP一定要為要訪問的DNS服務器的IP,可通過網絡設置查看
env.put(Context.PROVIDER_URL, "dns://10.17.45.239");
DirContext ctx = new InitialDirContext(env);
System.out.println("a:" + ctx);
DirContext ctx1 = (DirContext) ctx.lookup("www.sina.com");
System.out.println("b:" + ctx1);
printAttributes("c:", ctx1.getAttributes(""));
//從ctx.getAttributes("www.sina.com")與ctx1.getAttributes("")結果一樣
printAttributes("d:", ctx.getAttributes("www.sina.com"));
Attributes attrs1 = ctx.getAttributes("www.sina.com",
new String[] { "a" });
Attributes attrs2 = ctx.getAttributes("www.163.com",
new String[] { "a" });
Attributes attrs3 = ctx1.getAttributes("", new String[] { "a" });
Attributes attrs4 = ctx.getAttributes("www.baidu.com",
new String[] { "a" });
printAttributes("e:", attrs1);
printAttributes("f:", attrs2);
printAttributes("g:", attrs3);
printAttributes("attrs4:", attrs4);
System.out.println("nameParse:"+ctx1.getNameInNamespace());
//list,此方法會導致程序lock
//listEnumation("list:",ctx.list(""));
//----------------------search
Attributes matchAttrs = new BasicAttributes(true);
matchAttrs.put(new BasicAttribute("a", "61.172.201.13"));
NamingEnumeration answer = ctx1.search("www.sina.com", matchAttrs);
printNamingEnumeration("search :", answer);
}
public static void printAttributes(String tag, Attributes attres)
throws Exception {
for (NamingEnumeration ae = attres.getAll(); ae.hasMore();) {
Attribute attr = (Attribute) ae.next();
System.out
.println(tag
+ "-----------------------------------------------\nattribute: "
+ attr.getID());
/* Print each value */
for (NamingEnumeration e = attr.getAll(); e.hasMore(); System.out
.println("value: " + e.next()))
;
}
}
public static void listEnumation(String tag, NamingEnumeration name)
throws Exception {
for (; name.hasMore();) {
NameClassPair nameClass = (NameClassPair) name.next();
System.out
.println(tag
+ "-----------------------------------------------\nattribute: "
+ nameClass.getName() + ":"
+ nameClass.getClassName());
}
}
public static void printNamingEnumeration(String tag, NamingEnumeration e)
throws Exception {
for (; e.hasMore();) {
Attribute attr = (Attribute) e.next();
System.out
.println(tag
+ "-----------------------------------------------\nattribute: "
+ attr.getID());
/* Print each value */
for (NamingEnumeration ve = attr.getAll(); ve.hasMore(); System.out
.println("value: " + ve.next()))
;
}
}
}
上例中,在jdk1.4中可運行通過。對于DNS Service Provider更詳細的文檔,大家可通過此URL下載:http://java.sun.com/products/jndi/downloads/index.html
上例一個可能運行結果如下:
a:javax.naming.directory.InitialDirContext@1bf216a
b:com.sun.jndi.dns.DnsContext@3a6727
c:-----------------------------------------------
attribute: CNAME
value: us.sina.com.cn.
d:-----------------------------------------------
attribute: CNAME
value: us.sina.com.cn.
e:-----------------------------------------------
attribute: A
value: 218.30.66.67
value: 218.30.66.68
value: 218.30.66.69
value: 218.30.66.70
value: 218.30.66.71
value: 218.30.66.56
value: 218.30.66.57
value: 218.30.66.58
value: 218.30.66.59
value: 218.30.66.60
value: 218.30.66.61
value: 218.30.66.62
value: 218.30.66.63
value: 218.30.66.64
value: 218.30.66.65
value: 218.30.66.66
f:-----------------------------------------------
attribute: A
value: 220.181.28.42
g:-----------------------------------------------
attribute: A
value: 218.30.66.68
value: 218.30.66.69
value: 218.30.66.70
value: 218.30.66.71
value: 218.30.66.56
value: 218.30.66.57
value: 218.30.66.58
value: 218.30.66.59
value: 218.30.66.60
value: 218.30.66.61
value: 218.30.66.62
value: 218.30.66.63
value: 218.30.66.64
value: 218.30.66.65
value: 218.30.66.66
value: 218.30.66.67
attrs4:-----------------------------------------------
attribute: A
value: 220.181.27.5
nameParse:www.sina.com.
Exception in thread "main" javax.naming.OperationNotSupportedException
at com.sun.jndi.dns.DnsContext.c_search(Unknown Source)
at com.sun.jndi.toolkit.ctx.ComponentDirContext.p_search(Unknown Source)
at com.sun.jndi.toolkit.ctx.PartialCompositeDirContext.search(Unknown Source)
at com.sun.jndi.toolkit.ctx.PartialCompositeDirContext.search(Unknown Source)
at com.sun.jndi.toolkit.ctx.PartialCompositeDirContext.search(Unknown Source)
at com.sily.jndi.TestDNSJndi.main(TestDNSJndi.java:57)
示例分析:
通過分析代碼,我們可以看出我們從DNS服務器獲取了指定域名的IP地址,而且可以看出www.sina.com有多個IP.
另外,可以看出從ctx.getAttributes("www.sina.com")得到的結果與ctx1.getAttributes("")結果一樣,這便是目錄操作的兩種模式,這兩種模式取得的結果是一樣的,這點可以參考API文檔(http://java.sun.com/j2se/1.5.0/docs/api/javax/naming/directory/DirContext.html):
There are two basic models of what attributes should be associated with. First, attributes may be directly associated with a DirContext object. In this model, an attribute operation on the named object is roughly...
另外,還有一點需要注意,從ctx.getAttributes()方法返回的Attributes中包含多個Attribute,每個Attribute包含多個values,其它詳細內容請參考API文檔
最后,代碼NamingEnumeration answer = ctx1.search("www.sina.com", matchAttrs);試圖對ctx1進行屬性查找,但是拋出了異常,查看 DNS Service Provider 的文檔可知,DNS Service Provider 沒有提供對search方法的支持,大家可用其它的SP來測試此方法,如LDAP SP
總結:
此例只是簡單地演示的JNDI的目錄操作,對于目錄操作的其它高級主題如Search,Search Scope,Count Limit,Composite Names 等沒有詳細介紹,請參考其它相關文檔.