隨筆-4  評(píng)論-0  文章-0  trackbacks-0
            2006年7月7日

          最近的工作常常要和XML格式的文檔或字符串打交道,發(fā)現(xiàn)用JDOM來(lái)做真是方便。可以實(shí)現(xiàn)XML應(yīng)用程序的快速開(kāi)發(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 語(yǔ)言中使用 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)建
          ??? 我們由零開(kāi)始利用JDOM生成一個(gè)XML文檔。最后的結(jié)果(樣本文檔)看起來(lái)象這樣:
          ??? <?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)建的屬性添加到根元素。
          ??????? 這兩行代碼你也可以合成一行來(lái)寫,象這樣:
          ??????? 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í)慣,你完全可以分步來(lái)做,就象本節(jié)剛開(kāi)始的時(shí)候一樣。事實(shí)上如果層次比較多,寫成分步的形式更清晰些,也不容易出錯(cuò)。
          ??? 4.刪除子元素
          ??????? 這個(gè)操作比較簡(jiǎn)單:
          ??????? rootElement.removeChild("sex");//該方法返回一個(gè)布爾值
          ???????
          ??????? 到目前為止,我們學(xué)習(xí)了一下JDOM文檔生成操作。上面建立了一個(gè)樣本文檔,可是我們?cè)趺粗缹?duì)不對(duì)呢?因此需要輸出來(lái)看一下。我們將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è)專門的用來(lái)定義格式化輸出的類:org.jdom.output.Format,如果你沒(méi)有特殊的要求,有時(shí)候使用里面的幾個(gè)靜態(tài)方法(應(yīng)該可以說(shuō)是預(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)單說(shuō)明:
          ??????? getChild("childname") 返回指定名字的子節(jié)點(diǎn),如果同一級(jí)有多個(gè)同名子節(jié)點(diǎn),則只返回第一個(gè);如果沒(méi)有返回null值。
          ??????? getChildren("childname") 返回指定名字的子節(jié)點(diǎn)List集合。這樣你就可以遍歷所有的同一級(jí)同名子節(jié)點(diǎn)。
          ??????? getAttributeValue("name") 返回指定屬性名字的值。如果沒(méi)有該屬性則返回null,有該屬性但是值為空,則返回空字符串。
          ??????? getChildText("childname") 返回指定子節(jié)點(diǎn)的內(nèi)容文本值。
          ??????? getText() 返回該元素的內(nèi)容文本值。
          ???????
          ??????? 還有其他沒(méi)有羅列出來(lái)的方法,如果需要的話,可以隨時(shí)查閱JDOM的在線文檔:http://www.jdom.org/docs/apidocs/index.html。當(dāng)然你可以在你需要的地方添加、刪除元素操作,還記得上面的創(chuàng)建XML的方法嗎?呵呵~~~
          ???????
          ??????? 學(xué)習(xí)新東東還是從實(shí)例學(xué)起最為快捷,下面簡(jiǎn)單舉個(gè)例子,就以上面的XML樣本代碼來(lái)學(xué)習(xí)JDOM的XML解析。本例中讀取了樣本XML文件里一些屬性和content,最后我們還在contact元素里插入了一個(gè)新元素<email value="wanghua@cyberobject.com" />。盡管我們實(shí)現(xiàn)了對(duì)于XML的基本操作,細(xì)心的朋友可能會(huì)
          有疑問(wèn):如果XML文檔的層次稍微復(fù)雜一些,如果嵌套多達(dá)幾十上百層的話(開(kāi)個(gè)玩笑),如果靠這樣從根元素一級(jí)一級(jí)地通過(guò)getChild("childname")來(lái)訪問(wèn)子元素的話,將會(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();
          ??? }
          }
          }

          posted @ 2006-07-07 11:25 磐石 閱讀(329) | 評(píng)論 (0)編輯 收藏
            2006年7月2日
          使用DOM方式,Java解析XML基本步驟:
          首先,我們需要建立一個(gè)解析器工廠。
          DocumentBuilderFactory dbf=DocumentBuilderFactory.newInstance();
          然后可以利用這個(gè)工廠來(lái)獲得一個(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()方法來(lái)得列表中的每一個(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();
          }
          }
          }

          posted @ 2006-07-02 10:21 磐石 閱讀(3436) | 評(píng)論 (0)編輯 收藏
            2006年7月1日
          http://blog.chinaitlab.com/user1/270929/subject/1218.html
          posted @ 2006-07-01 15:30 磐石 閱讀(210) | 評(píng)論 (0)編輯 收藏
          ?
            其實(shí),簡(jiǎn)單的分析一下,就可以看出客戶和服務(wù)通訊的主要通道就是Socket本身,而服務(wù)器通過(guò)accept方法就是同意和客戶建立通訊.這樣當(dāng)客戶建立Socket的同時(shí)。服務(wù)器也會(huì)使用這一根連線來(lái)先后通訊,那么既然如此只要我們存在多條連線就可以了。那么我們的程序可以變?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)來(lái)就為它分配一個(gè)Socket直到這個(gè)客戶完成一次和服務(wù)器的交互,這里也就是接受到客戶的"End"消息.那么現(xiàn)在就實(shí)現(xiàn)了多客戶之間的交互了。但是.問(wèn)題又來(lái)了,這樣做雖然解決了多客戶,可是是排隊(duì)執(zhí)行的。也就是說(shuō)當(dāng)一個(gè)客戶和服務(wù)器完成一次通訊之后下一個(gè)客戶才可以進(jìn)來(lái)和服務(wù)器交互,無(wú)法做到同時(shí)服務(wù),那么要如何才能同時(shí)達(dá)到既能相互之間交流又能同時(shí)交流呢?很顯然這是一個(gè)并行執(zhí)行的問(wèn)題了。所以線程是最好的解決方案。

            那么下面的問(wèn)題是如何使用線程.首先要做的事情是創(chuàng)建線程并使得其可以和網(wǎng)絡(luò)連線取得聯(lián)系。然后由線程來(lái)執(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類繼承了下來(lái).并且通過(guò)構(gòu)造函數(shù)傳遞引用和客戶Socket建立了聯(lián)系,這樣每個(gè)線程就有了。一個(gè)通訊管道.同樣我們可以填寫run方法,把之前的操作交給線程來(lái)完成,這樣多客戶并行的Socket就建立起來(lái)了。

            以上的代碼使用的是

          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)該是很好的選擇了。
          posted @ 2006-07-01 09:38 磐石 閱讀(313) | 評(píng)論 (0)編輯 收藏
          僅列出標(biāo)題  
          主站蜘蛛池模板: 河间市| 金川县| 金门县| 拜城县| 河间市| 岗巴县| 惠来县| 鲁山县| 区。| 茂名市| 夏河县| 满城县| 万州区| 绩溪县| 易门县| 琼结县| 山阳县| 崇信县| 绿春县| 巧家县| 株洲市| 鄢陵县| 海阳市| 克山县| 三门县| 牡丹江市| 利津县| 木兰县| 黔东| 广州市| 武川县| 德钦县| 开鲁县| 鄢陵县| 阿拉尔市| 抚顺县| 清新县| 年辖:市辖区| 会理县| 会宁县| 威宁|