出自:yesky
我想您一定對XML有所了解,說不定您現(xiàn)在還躍躍欲試想寫一段XML文本呢,可是現(xiàn)在能找到的跨平臺的、免費的XML編輯器太少了。所以在本文中,我想介紹一下或者說帶您一步一步的開發(fā)一個簡單的XML編輯器,當(dāng)然我們要用到一些最常見的Java 2 Swing組件,不過這些都是免費的,有些是JDK中的,有些是可以從網(wǎng)上下載的。我想通過本文,你就可以創(chuàng)建一個屬于你自己的XML編輯器。 先讓我介紹一下本文輯寫的思路。首先我想簡要的討論一下XML和為什么樹型結(jié)構(gòu)比較適合用來顯示XML,然后我們來看一看JAXP API如何建立所需要的XML類的環(huán)境;然后我們將了解用來顯示一個圖形樹的JTree Swing組件;最后,我們將創(chuàng)建一個繼承JTree組件的可以重復(fù)使用的類,可以用來分析一個XML文檔,并把數(shù)據(jù)顯示在一個Jtree中。 說到XML(eXtensible Markup Languge),人們往往把它當(dāng)成是一種新的用于Web瀏覽器中的標(biāo)記語言,就象HTML或CSS一樣。其實,XML是一種數(shù)據(jù)表示語言,它允許你使用一種非常有效的方法來描述你的數(shù)據(jù)。XML能夠使你定義諸如“these three words constitutes a heading”這樣的語句。XML允許你聲明任何類型的數(shù)據(jù),而不是用來把這些數(shù)據(jù)顯示在網(wǎng)頁中。 請看一看下面的XML實例: <article> <header> <title> 使用Java Swing 創(chuàng)建一個XML編輯器 <subtitle> 第一部分</subtitle> </title> <author> Wayne </author> <header> <content> 這是正文</content> </article> 請注意,這些元素和標(biāo)準(zhǔn)的HTML語句是不同的,但是它們看上去比較象HTML,這是因為XML和HTML都是來源于SGML語言。不同的是HTML有預(yù)定義的標(biāo)簽集,而XML的語法則有許多靈活性,它允許你使用表意的標(biāo)記如<author>來括在數(shù)據(jù)兩邊。你還要注意,所有的元素都從屬于根元素(上例中為<article>),有些元素則還有自己的子元素,如<subtitle>就是<title>的子元素。這樣的數(shù)據(jù)組織方式有三個好處:數(shù)據(jù)能夠更加表意,數(shù)據(jù)更加易維護而且數(shù)據(jù)更加容易作為一個樹的結(jié)構(gòu)表現(xiàn)出來,這就是我們?yōu)槭裁词褂肑Tree對象來顯示XML數(shù)據(jù)的原因。如果你想對XML有更深的了解,請參閱天極網(wǎng)上的相關(guān)教程。 JAXP是一個用于處理XML的Java API,它能夠使應(yīng)用程序分析并且轉(zhuǎn)化XML文檔,它的功能有點象JDBC API,都是把函數(shù)功能抽象成一個個方法。你可以去Apache網(wǎng)站找到最新的Xerces分析器,其中含有最新的JAXP,下載下來以后把它放在你的類目錄中。 下面讓我們看一下如何使用JTree Swing組件。 我們都知道,在自然界中,一棵樹通常都有一個非常粗的樹干,樹干上有許多樹枝分叉。每個樹杈和樹杈之間都有一定的聯(lián)系,因為它們都有同一個來源:樹干。這種繼承的關(guān)系并不只在樹枝中有,人類譜系也遵循相同的規(guī)律。從父母,到子女再到子女的子女,就這樣直到數(shù)不清為止。同樣,在數(shù)據(jù)存儲中,樹的概念也是一種使用同人類家譜樹一樣方法儲存數(shù)據(jù)的方法。樹的每一個樹杈稱為一個節(jié)點,每個有子節(jié)點的節(jié)點稱為父節(jié)點,所有的子節(jié)點的公共的父節(jié)點被稱為根節(jié)點。一個JTree組件就是一個簡單的樹數(shù)據(jù)結(jié)構(gòu)的可視化表現(xiàn)形式。 幾乎所有的XML編輯器都包括一個可視化的樹結(jié)構(gòu),能讓你編輯XML文檔中的元素。我們馬上就會構(gòu)建一個編輯器,不過在此之前,先讓我們再了解一下JTree組件。一個節(jié)點在一棵樹的某個位置儲存數(shù)據(jù),為了存儲數(shù)據(jù),必須知道任何一個父節(jié)點和它們的子節(jié)點。javax.swing.tree包定義了一些非常有用的接口,提供了一種通用的方法構(gòu)建和操作一個樹結(jié)構(gòu)。 TreeNode方法,用于訪問樹的節(jié)點的信息 MutableTreeNode方法 用在一個可變的樹上(能夠添加或刪除子節(jié)點) TreeModel方法 用于創(chuàng)建和管理與樹有關(guān)的數(shù)據(jù)模型。 接下來,我們將創(chuàng)建一個繼承JTree的類,提供分析XML文檔和用可視化JTree組件把節(jié)點顯示出來的功能。 創(chuàng)建XTree組件 XTree類由一個構(gòu)造函數(shù)和三個方法組成,為了簡單起見我們的組件只能構(gòu)建一個Xtree,在樹創(chuàng)建好之后不能進行處理它的節(jié)點。下面讓我們來看一個這個類。 域: private DefaultMutableTreeNode treeNode 這個成員變量儲存TreeNode對象用于存儲JTree的模型?!?BR> DefaultMutableTreeNode類是在javax.swing.tree中被定義的,默認(rèn)提供了MutableTreeNode接口的一個實現(xiàn)。 private DocumentBuilderFactory dbf private DocumentBuilder db private Document doc 這三個成員變量是JAXP的一部分,用來分析XML文本并轉(zhuǎn)化成DOM(Document Object Model) 對象。 構(gòu)造函數(shù) public XTree( String text ) 這個構(gòu)造函數(shù)通過使用傳送到構(gòu)造器中的XML文本創(chuàng)建一個XTree對象。在初始化一些與JTree超類和DOM分析對象有關(guān)的基本顯示屬性后,構(gòu)造函數(shù)生成一個TreeModel 對象用來創(chuàng)建一個實際可視的樹。通過把DOM對象傳送到createTreeNode()方法來創(chuàng)建一個根節(jié)點,createTreeNode()方法返回一個DefaultMutableTreeNode類型的對象。這個對象然后被用來創(chuàng)建樹的TreeModel。 方法 private DefaultMutableTreeNode createTreeNode( Node root ) 這個方法采用一個DOM 節(jié)點,然后在子節(jié)點中遞歸直到所有的接點都被添加到DefaultMutableTreeNode中。這是一個遞歸方法,為了找到根節(jié)點下的每一個子節(jié)點,它每次都要調(diào)用自己。JTree然后就可以使用DefaultMutableTreeNode對象了,因為它已經(jīng)是樹型了。 private String getNodeType( Node node ) 這個方法,被createTreeNode()用來聯(lián)系一個字符串和某一種類型的節(jié)點。 private Node parseXml() 這個方法,用來分析XML文本字符串,它返回Node類型的對象,能夠被傳送到createTreeNode()方法中。 下面我給出了java代碼,供大家分析研究。 // 到入W3C的DOM 類 import org.w3c.dom.*; // JAXP的用于DOM I/O的類 import javax.xml.parsers.*; // 標(biāo)準(zhǔn)Java類 import javax.swing.*; import javax.swing.tree.*; import javax.swing.event.*; import java.awt.*; import java.awt.event.*; import java.io.*; public class XTree extends JTree { /** * 這個成員變量儲存TreeNode對象用于存儲JTree的模型。 *DefaultMutableTreeNode類是在javax.swing.tree中被定義的 *默認(rèn)提供了MutableTreeNode接口的一個實現(xiàn)。 */ private DefaultMutableTreeNode treeNode; /** * 這三個成員變量是JAXP的一部分,用來分析XML文本并轉(zhuǎn)化成DOM(Document Object Model) 對象。 */ private DocumentBuilderFactory dbf; private DocumentBuilder db; private Document doc; /** * 這個構(gòu)造函數(shù)通過使用傳送到構(gòu)造器中的XML文本創(chuàng)建一個XTree對象 * @參數(shù) text是一個XML格式的XML文本 * @異常 ParserConfigurationException 如果構(gòu)造函數(shù)非正常的設(shè)置分析器,就會拋出異常 */ public XTree( String text ) throws ParserConfigurationException { super(); // 設(shè)置Tree渲染的基本屬性 getSelectionModel().setSelectionMode( TreeSelectionModel.SINGLE_TREE_SELECTION ); setShowsRootHandles( true ); setEditable( false ); // 允許樹可以編輯 // 通過初始化對象的DOM來分析對象 dbf = DocumentBuilderFactory.newInstance(); dbf.setValidating( false ); db = dbf.newDocumentBuilder(); // 采用DOM根節(jié)點并且把它轉(zhuǎn)化成JTree的樹模型 treeNode = createTreeNode( parseXml( text ) ); setModel( new DefaultTreeModel( treeNode ) ); } file://中止XTree() /** * 這個方法采用一個DOM 節(jié)點,然后在子節(jié)點中遞歸直到所有的接點都被添加到DefaultMutableTreeNode中。 * 這是一個遞歸方法,為了找到根節(jié)點下的每一個子節(jié)點,它每次都要調(diào)用自己。 * JTree然后就可以使用DefaultMutableTreeNode對象了,因為它已經(jīng)是樹型了。 * * @參數(shù) root org.w3c.Node.Node * * @返回值 返回一個基于根節(jié)點DefaultMutableTreeNode對象 */ private DefaultMutableTreeNode createTreeNode( Node root ) { DefaultMutableTreeNode treeNode = null; String type, name, value; NamedNodeMap attribs; Node attribNode; // 從根節(jié)點中取得數(shù)據(jù) type = getNodeType( root ); name = root.getNodeName(); value = root.getNodeValue(); treeNode = new DefaultMutableTreeNode( root.getNodeType() == Node.TEXT_NODE ? value : name ); // 顯示屬性 attribs = root.getAttributes(); if( attribs != null ) { for( int i = 0; i < attribs.getLength(); i++ ) { attribNode = attribs.item(i); name = attribNode.getNodeName().trim(); value = attribNode.getNodeValue().trim(); if ( value != null ) { if ( value.length() > 0 ) { treeNode.add( new DefaultMutableTreeNode( "[Attribute] --> " + name + "=\"" + value + "\"" ) ); } file://end if ( value.length() > 0 ) } file://end if ( value != null ) } file://end for( int i = 0; i < attribs.getLength(); i++ ) } file://end if( attribs != null ) // 如果存在子節(jié)點,遞歸 if( root.hasChildNodes() ) { NodeList children; int numChildren; Node node; String data; children = root.getChildNodes(); // 如果子節(jié)點非空的話,只遞歸 if( children != null ) { numChildren = children.getLength(); for (int i=0; i < numChildren; i++) { node = children.item(i); if( node != null ) { if( node.getNodeType() == Node.ELEMENT_NODE ) { treeNode.add( createTreeNode(node) ); } file://end if( node.getNodeType() == Node.ELEMENT_NODE ) data = node.getNodeValue(); if( data != null ) { data = data.trim(); if ( !data.equals("\n") && !data.equals("\r\n") && data.length() > 0 ) { treeNode.add(createTreeNode(node)); } file://end if ( !data.equals("\n") && !data.equals("\r\n") && data.length() > 0 ) } file://end if( data != null ) } file://end if( node != null ) } file://end for (int i=0; i < numChildren; i++) } file://end if( children != null ) } file://end if( root.hasChildNodes() ) return treeNode; } file://end createTreeNode( Node root ) /** * 這個方法,被createTreeNode()用來聯(lián)系一個字符串和某一種類型的節(jié)點。 * * @參數(shù) node org.w3c.Node.Node * * @返回值 返回顯示節(jié)點類的字符串 */ private String getNodeType( Node node ) { String type; switch( node.getNodeType() ) { case Node.ELEMENT_NODE: { type = "Element"; break; } case Node.ATTRIBUTE_NODE: { type = "Attribute"; break; } case Node.TEXT_NODE: { type = "Text"; break; } case Node.CDATA_SECTION_NODE: { type = "CData section"; break; } case Node.ENTITY_REFERENCE_NODE: { type = "Entity reference"; break; } case Node.ENTITY_NODE: { type = "Entity"; break; } case Node.PROCESSING_INSTRUCTION_NODE: { type = "Processing instruction"; break; } case Node.COMMENT_NODE: { type = "Comment"; break; } case Node.DOCUMENT_NODE: { type = "Document"; break; } case Node.DOCUMENT_TYPE_NODE: { type = "Document type"; break; } case Node.DOCUMENT_FRAGMENT_NODE: { type = "Document fragment"; break; } case Node.NOTATION_NODE: { type = "Notation"; break; } default: { type = "???"; break; } }// 結(jié)束 switch( node.getNodeType() ) return type; } file://結(jié)束 getNodeType() /** * 這個方法,用來分析XML文本字符串,它返回Node類型的對象,能夠被傳送到createTreeNode()方法中。 * * @參數(shù) text 一個顯示XML文檔的字符串 * @返回值 返回一個org.w3c.Node.Node對象 */ private Node parseXml( String text ) { ByteArrayInputStream byteStream; byteStream = new ByteArrayInputStream( text.getBytes() ); try { doc = db.parse( byteStream ); } catch ( Exception e ) { e.printStackTrace(); System.exit(0); } return ( Node )doc.getDocumentElement(); } file://結(jié)束 parseXml() } file://結(jié)束 class XTree 代碼2 XTreeTester.java import javax.xml.parsers.*; // GUI類 import javax.swing.*; import java.awt.*; import java.awt.event.*; file://標(biāo)準(zhǔn) Java類 import java.io.*; public class XTreeTester extends JFrame { // XTree對象,用來在JTree中顯示XML private XTree xTree; // JScrollPane是JTree的容器 private JScrollPane jScroll; private WindowListener winClosing; // 設(shè)置框架的寬和高 private static final int FRAME_WIDTH = 400; private static final int FRAME_HEIGHT = 300; /* * 構(gòu)造器構(gòu)造一個框架包含JScrollPane, * 把一個基于XML字符串的XTree對象傳到構(gòu)造函數(shù)中 */ public XTreeTester( String title, String xml ) throws ParserConfigurationException { super( title ); Toolkit toolkit; Dimension dim, minimumSize; int screenHeight, screenWidth; // 初始化基本的布局屬性 setBackground( Color.lightGray ); getContentPane().setLayout( new BorderLayout() ); toolkit = Toolkit.getDefaultToolkit(); dim = toolkit.getScreenSize(); screenHeight = dim.height; screenWidth = dim.width; setBounds( (screenWidth-FRAME_WIDTH)/2, (screenHeight-FRAME_HEIGHT)/2, FRAME_WIDTH, FRAME_HEIGHT ); // 構(gòu)建XTree對象 xTree = new XTree( xml ); file://把XTree封裝到JScroll中,以便在JFrame可以使它在屏幕中上下滾動. jScroll = new JScrollPane(); jScroll.getViewport().add( xTree ); // 添加滾動條到框架中 getContentPane().add( jScroll, BorderLayout.CENTER ); validate(); setVisible(true); // 添加WindowListener用來關(guān)閉窗口 winClosing = new WindowAdapter() { public void windowClosing(WindowEvent e) { exit(); } }; addWindowListener(winClosing); } // 程序從這里開始執(zhí)行。必須把一個以xml為擴展名的XML文件傳送到這個方法中,其格式為java XTreeTester yourxmlfilename.xml public static void main( String[] args ) { String fileName = ""; BufferedReader reader; String line; StringBuffer xmlText; XTreeTester xTreeTester; // 創(chuàng)建一個基于特定XML文件的文檔對象 try { if( args.length > 0 ) { fileName = args[0]; if ( fileName.substring( fileName.indexOf( ‘.‘ ) ).equals( ".xml" ) ) { reader = new BufferedReader( new FileReader( fileName ) ); xmlText = new StringBuffer(); while ( ( line = reader.readLine() ) != null ) { xmlText.append( line ); } // 分析完文檔對象后將重寫文件 reader.close(); // 構(gòu)造 GUI 組件 xTreeTester = new XTreeTester( "XTree 測試", xmlText.toString() ); } else { help(); } } else { help(); } } catch( FileNotFoundException fnfEx ) { System.out.println( "沒有發(fā)現(xiàn)"+ fileName + "文件。" ); exit(); } catch( Exception ex ) { ex.printStackTrace(); exit(); } } file://幫助信息 private static void help() { System.out.println( "\n使用方法:java XTreeTester yourxmlfilename.xml" ); System.exit(0); } // 退出 private static void exit() { System.out.println( "\n謝謝使用 XTree" ); System.exit(0); } } |
|