
2006年7月7日
最近的工作常常要和XML格式的文檔或字符串打交道,發(fā)現(xiàn)用JDOM來做真是方便。可以實(shí)現(xiàn)XML應(yīng)用程序的快速開發(fā)。
??? 在 JDOM 中,XML 元素就是 Element 的實(shí)例,XML 屬性就是 Attribute 的實(shí)例,XML 文檔本身就是 Document 的實(shí)例。
??? 因?yàn)?JDOM 對(duì)象就是像 Document、Element 和 Attribute 這些類的直接實(shí)例,因此創(chuàng)建一個(gè)新 JDOM 對(duì)象就如在 Java 語言中使用 new 操作符一樣容易。JDOM 的使用是直截了當(dāng)?shù)摹?br />??? JDOM 使用標(biāo)準(zhǔn)的 Java 編碼模式。只要有可能,它使用 Java new 操作符而不故弄玄虛使用復(fù)雜的工廠化模式,使對(duì)象操作即便對(duì)于初學(xué)用戶也很方便。
???
??? 本文分兩步對(duì)JDOM的應(yīng)用加以介紹:XML創(chuàng)建 和 XML解析
一、XML文檔創(chuàng)建
??? 我們由零開始利用JDOM生成一個(gè)XML文檔。最后的結(jié)果(樣本文檔)看起來象這樣:
??? <?xml version="1.0" encoding="UTF-8"?>
??? <MyInfo comment="introduce myself">
??????? <name>kingwong</name>
??????? <sex value="male"/>
??????? <contact>
??????????? <telephone>87654321</telephone>
??????? </contact>
??? </MyInfo>
??? 1.以 MyInfo 為根元素創(chuàng)建文檔
??????? Element rootElement = new Element("MyInfo");//所有的XML元素都是 Element 的實(shí)例。根元素也不例外:)
??????? Document myDocument = new Document(rootElement);//以根元素作為參數(shù)創(chuàng)建Document對(duì)象。一個(gè)Document只有一個(gè)根,即root元素。
??? 2.給根元素添加屬性
??????? Attribute rootAttri = new Attribute("comment","introduce myself");//創(chuàng)建名為 commnet,值為 introduce myself 的屬性。
??????? rootElement.setAttribute(rootAttri);//將剛創(chuàng)建的屬性添加到根元素。
??????? 這兩行代碼你也可以合成一行來寫,象這樣:
??????? rootElement.setAttribute(new Attribute("comment","introduce myself"));
??????? 或者
??????? rootElement.setAttribute("comment","introduce myself");
??? 3.添加元素和子元素
??????? JDOM里子元素是作為 content(內(nèi)容)添加到父元素里面去的,所謂content就是類似上面樣本文檔中<name></name>之間的東東,即kingwong。羅嗦了點(diǎn)是吧:)
??????? Element nameElement = new Element("name");//創(chuàng)建 name 元素
??????? nameElement.addContent("kingwong");//將kingwong作為content添加到name元素
?rootElement.addContent(nameElement);//將name元素作為content添加到根元素
?
?這三行你也可以合為一句,象這樣:
?rootElement.addContent((Content)(new Element("name").addContent("kingwong")));//因?yàn)閍ddContent(Content child)方法返回的是一個(gè)Parent接口,而Element類同時(shí)繼承了Content類和實(shí)現(xiàn)了Parent接口,所以我們把它造型成Content。
?
??????? 我們用同樣的方法添加帶屬性的子元素<sex value="male"/>
??????? rootElement.addContent(new Element("sex").setAttribute("value","male"));//注意這里不需要轉(zhuǎn)型,因?yàn)閍ddAttribute(String name,String value)返回值就是一個(gè) Element。
???????
??????? 同樣的,我們添加<contract />元素到根元素下,用法上一樣,只是稍微復(fù)雜了一些:
??????? rootElement.addContent((Content)(new Element("contact").addContent((Content)(new Element("telephone").addContent("87654321")))));
??????? 如果你對(duì)這種簡(jiǎn)寫形式還不太習(xí)慣,你完全可以分步來做,就象本節(jié)剛開始的時(shí)候一樣。事實(shí)上如果層次比較多,寫成分步的形式更清晰些,也不容易出錯(cuò)。
??? 4.刪除子元素
??????? 這個(gè)操作比較簡(jiǎn)單:
??????? rootElement.removeChild("sex");//該方法返回一個(gè)布爾值
???????
??????? 到目前為止,我們學(xué)習(xí)了一下JDOM文檔生成操作。上面建立了一個(gè)樣本文檔,可是我們?cè)趺粗缹?duì)不對(duì)呢?因此需要輸出來看一下。我們將JDOM生成的文檔輸出到控制臺(tái),使用 JDOM 的 XMLOutputter 類。
??? 5.? 將 JDOM 轉(zhuǎn)化為 XML 文本
??????? XMLOutputter xmlOut = new XMLOutputter("? ",true);
?try {
? xmlOut.output(myDocument,System.out);
?} catch (IOException e) {
? e.printStackTrace();
?}
?XMLOutputter 有幾個(gè)格式選項(xiàng)。這里我們已指定希望子元素從父元素縮進(jìn)兩個(gè)空格,并且希望元素間有空行。
?new XMLOutputter(java.lang.String indent, boolean newlines)這個(gè)方法在最新版本中已經(jīng)不建議使用。JDOM有一個(gè)專門的用來定義格式化輸出的類:org.jdom.output.Format,如果你沒有特殊的要求,有時(shí)候使用里面的幾個(gè)靜態(tài)方法(應(yīng)該可以說是預(yù)定義格式)如 getPrettyFormat()就可以了。我們把上面的輸出格式稍微改一下,就象這樣:
?XMLOutputter xmlOut = new XMLOutputter(Format.getPrettyFormat());?
??? 6.將JDOM文檔轉(zhuǎn)化為其他形式
??????? XMLOutputter 還可輸出到 Writer 或 OutputStream。為了輸出JDOM文檔到一個(gè)文本文件,我們可以這樣做:
??????? FileWriter writer = new FileWriter("/some/directory/myFile.xml");
??????? outputter.output(myDocument, writer);
??????? writer.close();
???????
??????? XMLOutputter 還可輸出到字符串,以便程序后面進(jìn)行再處理:
??????? Strng outString = xmlOut.outputString(myDocument);
???????
??????? 當(dāng)然,在輸出的時(shí)候你不一定要輸出所有的整個(gè)文檔,你可以選擇元素進(jìn)行輸出:
??????? xmlOut.output(rootElement.getChild("name"),System.out);
??????? 一句話,JDOM非常靈活方便!如果你想進(jìn)一步研究JDOM,請(qǐng)到官方網(wǎng)站去看一看:http://www.jdom.org
??? 本節(jié)示例源碼:
package com.cyberobject.study;
import java.io.IOException;
import org.jdom.Attribute;
import org.jdom.Content;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.output.Format;
import org.jdom.output.XMLOutputter;
/**
?* @author kingwong
?*
?* TODO To change the template for this generated type comment go to
?* Window - Preferences - Java - Code Style - Code Templates
?*/
public class TestJDOM {
?public static void main(String[] args)
?{
? Element rootElement = new Element("MyInfo");
? Document myDocument = new Document(rootElement);
?
//? Attribute rootAttri = new Attribute("comment","introduce myself");
//? rootElement.setAttribute(rootAttri);
?
? rootElement.setAttribute("comment","introduce myself");
? //rootElement.setAttribute(new Attribute("comment","introduce myself"));
//? Element sexElement = new Element("sex");
//? rootElement.addContent(sexElement);
?
//? Element nameElement = new Element("name");
//? nameElement.addContent("kingwong");
//? rootElement.addContent(nameElement);
?
? rootElement.addContent((Content)(new Element("name").addContent("kingwong")));
? rootElement.addContent(new Element("sex").setAttribute("value","male"));
? rootElement.addContent((Content)(new Element("contract").addContent((Content)(new Element("telephone").addContent("87654321")))));
?
? rootElement.removeChild("sex");
?
? XMLOutputter xmlOut = new XMLOutputter(Format.getPrettyFormat());
? try {
?? xmlOut.output(myDocument,System.out);
?? //xmlOut.output(rootElement.getChild("name"),System.out);
?? //String outString = xmlOut.outputString(myDocument);
? } catch (IOException e) {
?? e.printStackTrace();
? }
?}
}
???????
二、XML文檔解析
??? JDOM 不光可以很方便的建立XML文檔,它的另一個(gè)用處是它能夠讀取并操作現(xiàn)有的 XML 數(shù)據(jù)。
??? JDOM的解析器在org.jdom.input.*這個(gè)包里,其中的DOMBuilder的功能是將DOM模型的Document解析成JDOM模型的Document;SAXBuilder的功能是從文件或流中解析出符合JDOM模型的XML樹。由于我們經(jīng)常要從一個(gè)文件里讀取數(shù)據(jù),因此我們應(yīng)該采用后者作為解析工具。
解析一個(gè)xml文檔,基本可以看成以下幾個(gè)步驟:
??? 1.實(shí)例化一個(gè)合適的解析器對(duì)象
??????? 本例中我們使用SAXBuilder:
??????? SAXBuilder sb = new SAXBuilder();
??? 2.以包含XML數(shù)據(jù)的文件為參數(shù),構(gòu)建一個(gè)文檔對(duì)象myDocument
??????? Document myDocument = sb.build(/some/directory/myFile.xml);
??? 3.獲到根元素
??????? Element rootElement = myDocument.getRootElement();
???????
??????? 一旦你獲取了根元素,你就可以很方便地對(duì)它下面的子元素進(jìn)行操作了,下面對(duì)Element對(duì)象的一些常用方法作一下簡(jiǎn)單說明:
??????? getChild("childname") 返回指定名字的子節(jié)點(diǎn),如果同一級(jí)有多個(gè)同名子節(jié)點(diǎn),則只返回第一個(gè);如果沒有返回null值。
??????? getChildren("childname") 返回指定名字的子節(jié)點(diǎn)List集合。這樣你就可以遍歷所有的同一級(jí)同名子節(jié)點(diǎn)。
??????? getAttributeValue("name") 返回指定屬性名字的值。如果沒有該屬性則返回null,有該屬性但是值為空,則返回空字符串。
??????? getChildText("childname") 返回指定子節(jié)點(diǎn)的內(nèi)容文本值。
??????? getText() 返回該元素的內(nèi)容文本值。
???????
??????? 還有其他沒有羅列出來的方法,如果需要的話,可以隨時(shí)查閱JDOM的在線文檔:http://www.jdom.org/docs/apidocs/index.html。當(dāng)然你可以在你需要的地方添加、刪除元素操作,還記得上面的創(chuàng)建XML的方法嗎?呵呵~~~
???????
??????? 學(xué)習(xí)新東東還是從實(shí)例學(xué)起最為快捷,下面簡(jiǎn)單舉個(gè)例子,就以上面的XML樣本代碼來學(xué)習(xí)JDOM的XML解析。本例中讀取了樣本XML文件里一些屬性和content,最后我們還在contact元素里插入了一個(gè)新元素<email value="wanghua@cyberobject.com" />。盡管我們實(shí)現(xiàn)了對(duì)于XML的基本操作,細(xì)心的朋友可能會(huì)
有疑問:如果XML文檔的層次稍微復(fù)雜一些,如果嵌套多達(dá)幾十上百層的話(開個(gè)玩笑),如果靠這樣從根元素一級(jí)一級(jí)地通過getChild("childname")來訪問子元素的話,將會(huì)非常痛苦!是的,的確是這樣,但是我們有另一個(gè)有力的工具XPath,為什么不用呢?這是后話!先賣個(gè)關(guān)子(手敲累啦,下回吧,呵呵)。
???????
/*
?* Created on 2004-8-21
?*
?* TODO To change the template for this generated file go to
?* Window - Preferences - Java - Code Style - Code Templates
?*/
package com.cyberobject.study;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.input.SAXBuilder;
import org.jdom.output.Format;
import org.jdom.output.XMLOutputter;
/**
?* @author kingwong
?*
?* TODO To change the template for this generated type comment go to
?* Window - Preferences - Java - Code Style - Code Templates
?*/
public class TestJDOM2 {
?public static void main(String[] args){
?SAXBuilder sb = new SAXBuilder();
??? try
??? {???????
???? Document doc = sb.build("myFile.xml");
? Element root = doc.getRootElement();
?
? String str1 = root.getAttributeValue("comment");
? System.out.println("Root Element's comment attribute is : " + str1);
? String str2 = root.getChild("sex").getAttributeValue("value");
? System.out.println("sex Element's value attribute is : " + str2);
? String str3 = root.getChildText("name");
? System.out.println("name Element's content is :" + str3);
? String str4 = root.getChild("contact").getChildText("telephone");
? System.out.println("contact Element's telephone subelement content is : " + str4 + "\n");
? Element inputElement = root.getChild("contact");
? inputElement.addContent(new Element("email").setAttribute("value","wanghua@cyberobject.com"));
?
? XMLOutputter xmlOut = new XMLOutputter(Format.getPrettyFormat());
???? String outStr = xmlOut.outputString(root);
???? System.out.println(outStr);
??? }
??? catch(Exception e)
??? {
??????? e.printStackTrace();
??? }
}
}

2006年7月2日
使用DOM方式,Java解析XML基本步驟:
首先,我們需要建立一個(gè)解析器工廠。
DocumentBuilderFactory dbf=DocumentBuilderFactory.newInstance();
然后可以利用這個(gè)工廠來獲得一個(gè)具體的解析對(duì)象。
DocumentBuilder builder=dbf.newDocumentBuilder();
DocumentBuilder的Parse()方法接受一個(gè)XML文檔名作為輸入?yún)?shù),返回一個(gè)Document對(duì)象。Document對(duì)象代表了 一個(gè)XML文檔的樹模型。
Document doc=builder.parse("candiate.xml");
使用Document對(duì)象的getElementsByTagName()方法,我們可以得到一個(gè)NodeList對(duì)象,他是XML文檔中的標(biāo)簽元素 列表,可以使用NodeList對(duì)象的item()方法來得列表中的每一個(gè)Node對(duì)象。
NodeList nl=doc.getElementsByTagName("PERSON");
Element node=(Element)nl.item(i);
最后,我們會(huì)使用Node對(duì)象的getNodeValue()方法提取某個(gè)標(biāo)簽內(nèi)的內(nèi)容。
node.getElementsByTagName("NAME").item(0).getFirstChild().getNodeValue()
完整程序代碼:
import javax.xml.parsers.*;
import org.w3c.dom.*;
public class dom {
public static void main(String args[]){
String uri=args[0];
try{
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();//建立一個(gè)解析器工廠。
DocumentBuilder builder=factory.newDocumentBuilder();//獲得一個(gè)具體的解析對(duì)象。
Document doc=builder.parse(uri);//返回一個(gè)Document對(duì)象。
System.out.println(doc.getImplementation());
NodeList nl =doc.getElementsByTagName("PERSON");//得到一個(gè)NodeList對(duì)象。
for (int i=0;i<nl.getLength();i++){
Element node=(Element) nl.item(i);//得列表中的每一個(gè)Node對(duì)象。
System.out.print("NAME: ");
System.out.println (node.getElementsByTagName("NAME").item(0).getFirstChild().getNodeValue());
System.out.print("ADDRESS: ");
System.out.println (node.getElementsByTagName("ADDRESS").item(0).getFirstChild().getNodeValue());
System.out.print("TEL: ");
System.out.println (node.getElementsByTagName("TEL").item(0).getFirstChild().getNodeValue());
System.out.print("FAX: ");
System.out.println (node.getElementsByTagName("FAX").item(0).getFirstChild().getNodeValue());
System.out.print("EMAIL: ");
System.out.println (node.getElementsByTagName("EMAIL").item(0).getFirstChild().getNodeValue());
System.out.println();
}
}catch(Exception e){
e.printStackTrace();
}
}
}

2006年7月1日
http://blog.chinaitlab.com/user1/270929/subject/1218.html
- ?
- 作者: ∣來源:天極java∣原文地址∣2006-6-28
-
其實(shí),簡(jiǎn)單的分析一下,就可以看出客戶和服務(wù)通訊的主要通道就是Socket本身,而服務(wù)器通過accept方法就是同意和客戶建立通訊.這樣當(dāng)客戶建立Socket的同時(shí)。服務(wù)器也會(huì)使用這一根連線來先后通訊,那么既然如此只要我們存在多條連線就可以了。那么我們的程序可以變?yōu)槿缦?
服務(wù)器:
import java.io.*; import java.net.*;
public class MyServer { public static void main(String[] args) throws IOException{ ServerSocket server=new ServerSocket(5678); while(true){ Socket client=server.accept(); BufferedReader in=new BufferedReader(new InputStreamReader(client.getInputStream())); PrintWriter out=new PrintWriter(client.getOutputStream()); while(true){ String str=in.readLine(); System.out.println(str); out.println("has receive...."); out.flush(); if(str.equals("end")) break; } client.close(); } } } |
這里僅僅只是加了一個(gè)外層的While循環(huán),這個(gè)循環(huán)的目的就是當(dāng)一個(gè)客戶進(jìn)來就為它分配一個(gè)Socket直到這個(gè)客戶完成一次和服務(wù)器的交互,這里也就是接受到客戶的"End"消息.那么現(xiàn)在就實(shí)現(xiàn)了多客戶之間的交互了。但是.問題又來了,這樣做雖然解決了多客戶,可是是排隊(duì)執(zhí)行的。也就是說當(dāng)一個(gè)客戶和服務(wù)器完成一次通訊之后下一個(gè)客戶才可以進(jìn)來和服務(wù)器交互,無法做到同時(shí)服務(wù),那么要如何才能同時(shí)達(dá)到既能相互之間交流又能同時(shí)交流呢?很顯然這是一個(gè)并行執(zhí)行的問題了。所以線程是最好的解決方案。
那么下面的問題是如何使用線程.首先要做的事情是創(chuàng)建線程并使得其可以和網(wǎng)絡(luò)連線取得聯(lián)系。然后由線程來執(zhí)行剛才的操作,要?jiǎng)?chuàng)建線程要么直接繼承Thread要么實(shí)現(xiàn)Runnable接口,要建立和Socket的聯(lián)系只要傳遞引用就可以了.而要執(zhí)行線程就必須重寫run方法,而run方法所做的事情就是剛才單線程版本main所做的事情,因此我們的程序變成了這樣:
import java.net.*; import java.io.*;
public class MultiUser extends Thread{ private Socket client;
public MultiUser(Socket c){ this.client=c; }
public void run(){ try{ BufferedReader in=new BufferedReader(new InputStreamReader(client.getInputStream())); PrintWriter out=new PrintWriter(client.getOutputStream()); //Mutil User but can parallel while(true){ String str=in.readLine(); System.out.println(str); out.println("has receive...."); out.flush(); if(str.equals("end")) break; } client.close(); }catch(IOException ex){ }finally{ } }
public static void main(String[] args)throws IOException{ ServerSocket server=new ServerSocket(5678); while(true){ //transfer location change Single User or Multi User MultiUser mu=new MultiUser(server.accept()); mu.start(); } } } |
我的類直接從Thread類繼承了下來.并且通過構(gòu)造函數(shù)傳遞引用和客戶Socket建立了聯(lián)系,這樣每個(gè)線程就有了。一個(gè)通訊管道.同樣我們可以填寫run方法,把之前的操作交給線程來完成,這樣多客戶并行的Socket就建立起來了。
以上的代碼使用的是
BufferedReader in=new BufferedReader(new InputStreamReader(client.getInputStream())); PrintWriter out=new PrintWriter(client.getOutputStream()); |
還有一種方法是使用
DataInputStream isFromClient = new DataInputStream(client.getInputStream()); DataOutputStream osToClient = new DataOutputStream(client.getOutputStream()); |
關(guān)于這兩種輸入輸出流的不同,我也只知道前一種對(duì)字符串支持比較好,后面對(duì)于讀取一個(gè)字符串需要處理,但是可以支持很多種類型的輸出。對(duì)于傳遞字符串而言前一種應(yīng)該是很好的選擇了。