|         目前,XML已经广泛应用于数据交换领域,xml是基于纯文本的,具有优秀的跨平台特性。本文将讲述如何在J2ME中解析XML进行数据传输。
         XML写法比较简单,而且非常便于人来阅读。但是必须在客户端和服务器端有XML的解析器才可以正常通信,由于初期的移动信息设备内存和处理器上的不足,所以在MIDP1.0中并没有提供对XML的支持。随着内存和处理器的提高,对XML的支持成为了可能。在JSR182中提供了XML的解析器,但是这并不是标准MIDP中的API需要特定的实现才可以支持。幸运的是有第三方的API对解析xml提供了支持,比较有名的是kxml和nanoxml。 
        使用XML之前,你必须考虑好是不是必须用它来传输数据,因为解析xml是比较耗费资源的,尤其是在CPU和内存的资源都很宝贵的条件下。如果我们能使用DataInputStream和DataOutputStream传输的话就尽量不要使用XML。XML的解析器有两种,一种是确认性的,他在解析之前会对xml的文档进行有效性的验证,确保这是应用程序需要的。另一种则是非确认性的,他不做验证工作直接进行解析,无疑这样的速度会快。kxml和nanoxml都是这样的解析器。它们也存在差别,kxml是增量解析器他会一点一点的解析,这样在解析大的文档的时候会效率比较高。nanoxml是一步解析器,一次就把文档解析完,如果文档很大的话,这无疑会耗费很大的内存。要使用kxml你可以从http://www.kxml.org 下载得到,在http://nanoxml.sourceforge.net 你可以下载nanoxml。 
        要使用kxml,你必须首先得到一个XmlParser实例,它用Reader作为构造器的参数: try {     Reader    r = .....;     XmlParser parser = new XmlParser( r ); } catch( java.io.IOException e ){     // handle exception.... } 
如果你的xml文档存储在String内的话你可以使用ByteArrayInputStream和InputStreamReader: String    xml = "<a>some xml</a>"; ByteArrayInputStream bin =              new ByteArrayInputStream( xml.getBytes() ); XmlParser parser = new XmlParser( new InputStreamReader( bin ) ); 当从网上接收数据的时候可以这样: HttpConnection    conn = .....; InputStreamReader doc =           new InputStreamReader( conn.openInputStream() ); XmlParser parser = new XmlParser( doc ); 得到parser实例后我们就可以调用read方法进行解析了,read方法会返回一个ParseEvent,通过判断他的类型我们就可以解析xml了。 try {     boolean keepParsing = true;              while( keepParsing ){         ParseEvent event = parser.read();           switch( event.getType() ){             case Xml.START_TAG:                 ..... // handle start of an XML tag                 break;             case Xml.END_TAG:                 ..... // handle end of an XML tag                 break;             case Xml.TEXT:                 ..... // handle text within a tag                 break;             case Xml.WHITESPACE:                 ..... // handle whitespace                 break;             case Xml.COMMENT:                 ..... // handle comment                 break;             case Xml.PROCESSING_INSTRUCTION:                 ..... // handle XML PI                 break;             case Xml.DOCTYPE:                 ..... // handle XML doctype                 break;             case Xml.END_DOCUMENT:                 ..... // end of document;                 keepParsing = false;                 break;         }     } } catch( java.io.IOException e ){ } 
如果想使用nanoxml,那么你首先要创建一个kXMLElement实例,然后调用parseFromReader、parseString或者parseCharArray。由于他是一步解析器,那么它会把整个文档解析完后生成一个Object tree。每个节点都是一个kXMLElement的实例,通过调用getChildren等方法可以在这棵树上导航。 HttpConnection    conn = .....; InputStreamReader doc =           new InputStreamReader( conn.openInputStream() ); kXMLElement       root = new kXMLElement(); 
try {     root.parseFromReader( doc ); } catch( kXMLParseException pe ){ } catch( IOException ie ){ } 
下面是一个J2ME的应用程序简单演示了如何解析xml。如果有时间可以写写复杂的测试程序。你可以从如下地址下载源代码:XMLTest. 里面包括了kxml和nanoxml的源代码,如果想得到最新的源代码请参考他们的官方网站,在本站提供了kxml的在线API package com.ericgiguere.techtips; 
import java.io.*; import java.util.*; import javax.microedition.lcdui.*; import javax.microedition.midlet.*; import nanoxml.*; import org.kxml.*; import org.kxml.parser.*; 
/**  * Simple MIDlet that demonstrates how an XML document can be  * parsed using kXML or NanoXML.  */ 
public class XMLTest extends MIDlet { 
    // Our XML document -- normally this would be something you     // download.          private static String xmlDocument =         "<list><item>apple</item>" +               "<item>orange</item>" +               "<item>pear</item></list>";                    private Display display;     private Command exitCommand = new Command( "Exit",                                                Command.EXIT, 1 );                                                     public XMLTest(){     }          protected void destroyApp( boolean unconditional )                        throws MIDletStateChangeException {         exitMIDlet();     }          protected void pauseApp(){     }          protected void startApp() throws MIDletStateChangeException {         if( display == null ){ // first time called...             initMIDlet();         }     }          private void initMIDlet(){         display = Display.getDisplay( this );                  String [] items;                  //items = parseUsingNanoXML( xmlDocument );         items = parseUsingkXML( xmlDocument );                  display.setCurrent( new ItemList( items ) );     }          public void exitMIDlet(){         notifyDestroyed();     }          // Parses a document using NanoXML, looking for     // "item" nodes and returning their content as an     // array of strings.          private String[] parseUsingNanoXML( String xml ){         kXMLElement root = new kXMLElement();         try {             root.parseString( xml );                          Vector list = root.getChildren();             Vector items = new Vector();                          for( int i = 0; i < list.size(); ++i ){                 kXMLElement node =                       (kXMLElement) list.elementAt( i );                 String      tag = node.getTagName();                                  if( tag == null ) continue;                 if( !tag.equals( "item" ) ) continue;                                  items.addElement( node.getContents() );             }                          String[] tmp = new String[ items.size() ];             items.copyInto( tmp );             return tmp;         }         catch( kXMLParseException ke ){             return new String[]{ ke.toString() };         }     }          // Parses a document using kXML, looking for "item"     // nodes and returning their content as an     // array of strings.          private String[] parseUsingkXML( String xml ){         try {             ByteArrayInputStream bin =                             new ByteArrayInputStream(                                       xml.getBytes() );             InputStreamReader in = new InputStreamReader( bin );             XmlParser parser = new XmlParser( in );             Vector    items = new Vector();                          parsekXMLItems( parser, items );                          String[] tmp = new String[ items.size() ];             items.copyInto( tmp );             return tmp;         }         catch( IOException e ){             return new String[]{ e.toString() };         }     }          private void parsekXMLItems( XmlParser parser, Vector items )                                      throws IOException {         boolean inItem = false;                  while( true ){             ParseEvent event = parser.read();             switch( event.getType() ){                 case Xml.START_TAG:                     if( event.getName().equals( "item" ) ){                         inItem = true;                     }                     break;                 case Xml.END_TAG:                     if( event.getName().equals( "item" ) ){                         inItem = false;                     }                     break;                 case Xml.TEXT:                     if( inItem ){                         items.addElement( event.getText() );                     }                     break;                 case Xml.END_DOCUMENT:                     return;             }         }     }          // Simple List UI component for displaying the list of     // items parsed from the XML document.          class ItemList extends List implements CommandListener {              ItemList( String[] list ){             super( "Items", IMPLICIT, list, null );             addCommand( exitCommand );             setCommandListener( this );         }                  public void commandAction( Command c, Displayable d ){             if( c == exitCommand ){                 exitMIDlet();             }          }     } } 
 
  
 
  |