(From: http://blog.csdn.net/ichliebephone/article/details/5971533)
一.基础知识
在《Android XML解析学习——Sax方式》中我们学习了Android上使用Java的SAX进行XML解析的方式。而在Android平台上使用SAX解析XML,除了使用Java的API外,还可以使用Android SDK带的 ="EN-US">API来实现。Android SDK中和SAX解析相关的包为android.sax,在这个包中Android提供了都有的SAX API,使用它们可以更加方便的进行SAX方式的XML解析。
当xml文件中在不同的位置处有相同的元素标签名时,在相应的事件回调处理函数中往往就需要进行判断处理。比如USGS的xml形式的地震数据为:
- xml version="1.0"?>
- <feed xmlns="" xmlns:georss="">
- <updated>2010-09-15T04:41:18Zupdated>
- <title>USGS M2.5+ Earthquakestitle>
- <subtitle>Real-time, worldwide earthquake list for the past daysubtitle>
- <link rel="self" href="" mce_href=""/>
- <link href="" mce_href=""/>
- <author><name>U.S. Geological Surveyname>author>
- <id>id>
- <icon>/favicon.icoicon>
- <entry>
- <id>urn:earthquake-usgs-gov:ak:10078833id>
- <title>M 2.9, Southern Alaskatitle>
- <updated>2010-09-15T04:14:03Zupdated>
- <link rel="alternate" type="text/html" href="recenteqsww/Quakes/ak10078833.php" mce_href="recenteqsww/Quakes/ak10078833.php"/>
- <summary type="html">
-
- summary>
- <georss:point>59.9094 -153.1241georss:point>
- <georss:elev>-98900georss:elev>
- <category label="Age" term="Past hour"/>
- entry>
- <entry>
-
- entry>
- feed>
我们看到在开始前就有、和等元素标签,而元素中也包含这些标签名,在SAX解析时都会产生相应的事件,但我们实际需要处理的为元素中的这些标签产生的事件,因此我们设置了一个变量
- private Boolean startEntryElementFlag = false;
来进行判断。对以上的xml数据来说,这样的处理还不会出现问题,但是如果需要解析一个更加复杂的XML文档,则类似的需要对不同位置处的相同标签名进行判断这样的处理可能会带来各种各样的Bug。
而使用android.sax包中的API来进行SAX方式的解析则不会有以上的问题。实际上使用android.sax包还有点结合了我们以后会详细介绍的DOM方式,首先获取需要解析部分的根元素,然后使用getChild方法获取具体的某个子元素,之后就可以为具体的元素设置事件处理的回调函数,比如为一个元素分别设置元素开始的事件处理setStartElementListener,元素包含的文本内容结束的事件处理setEndTextElementListener和元素结束的事件处理setEndElementListener。
概括的来说,android.sax包进行XML解析的过程为用DOM的方式获取具体位置处的元素,然后为其设置需要的事件处理函数。具体的实现我们可以看实例开发部分的代码。
Android SDK中提供的和XML相关的还有一个类:android.util.Xml,在该类中提供了比较实用的XML相关的类方法,比如开始解析的parse方法,和直接创建XmlPullParser及XmlSerializer(这两块内容以后介绍)的方法等。
下面我们就用上面介绍的Android SDK中的SAX方式来实现解析XML形式的USGS地震数据的Demo例子。
二.实例开发
我们要完成的效果图如下图1所示:
图1 ListView列表显示的地震数据
和上一部分Demo例子的一样,也是解析完地震数据后用ListView列表的方式显示每条地震的震级和地名信息。
新建一个Android工程AndroidXMLDemoSaxII。
添加进上一个Demo工程AndroidXMLDemoSax中的EarthquakeEntry.java文件,如果需要从本地读取xml数据的话,同时在assets文件夹下添加保存为xml格式了的USGS地震数据USGS_Earthquake_1M2_5.xml,如果需要联网读取的话,在manifest.xml文件中添加权限:
- "android.permission.INTERNET" />
并修改res/layout下的main.xml为:
- xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android=""
- android:orientation="vertical"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- >
- <ListView
- android:id="@+id/list"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- />
- LinearLayout>
接下来就来新建添加一个类AndroidSaxEarthquakeHandler,以Android SDK提供的包android.sdk的API来完成解析地震数据的具体逻辑实现,内容如下:
- public class AndroidSaxEarthquakeHandler {
-
- private String kRootElementName = "feed";
- private String kEntryElementName = "entry";
- private String kLinkElementName = "link";
- private String kTitleElementName = "title";
- private String kUpdatedElementName = "updated";
- private String kGeoRSSPointElementName = "point";
- private String kGeoRSSElevElementName = "elev";
- static final String ATOM_NAMESPACE = "";
- static final String GEORSS_NAMESPACE = "";
-
- private ArrayList earthquakeEntryList;
- private EarthquakeEntry earthquakeEntry;
-
-
- public ArrayList parse(InputStream inStream)
- {
- earthquakeEntryList = new ArrayList();
- RootElement root = new RootElement(ATOM_NAMESPACE, kRootElementName);
- Element entry = root.getChild(ATOM_NAMESPACE, kEntryElementName);
-
-
- entry.setStartElementListener(new StartElementListener() {
- @Override
- public void start(Attributes attributes) {
-
- earthquakeEntry = new EarthquakeEntry();
- }
- });
- entry.setEndElementListener(new EndElementListener() {
- @Override
- public void end() {
-
- earthquakeEntryList.add(earthquakeEntry);
- }
- });
-
- entry.getChild(ATOM_NAMESPACE, kTitleElementName).setEndTextElementListener(new EndTextElementListener() {
- @Override
- public void end(String currentData) {
-
-
- String magnitudeString = currentData.split(" ")[1];
- int end = magnitudeString.length()-1;
- magnitudeString = magnitudeString.substring(0, end);
- double magnitude = Double.parseDouble(magnitudeString);
- earthquakeEntry.setMagnitude(magnitude);
-
- String place = currentData.split(",")[1].trim();
- earthquakeEntry.setPlace(place);
- }
- });
-
- entry.getChild(ATOM_NAMESPACE, kUpdatedElementName).setEndTextElementListener(new EndTextElementListener() {
- @Override
- public void end(String currentData) {
-
-
- SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'");
- Date qdate = new GregorianCalendar(0,0,0).getTime();
- try {
- qdate = sdf.parse(currentData);
- } catch (ParseException e) {
- e.printStackTrace();
- }
- earthquakeEntry.setDate(qdate);
- }
- });
-
- entry.getChild(GEORSS_NAMESPACE, kGeoRSSPointElementName).setEndTextElementListener(new EndTextElementListener() {
- @Override
- public void end(String currentData) {
-
-
- String[] latLongitude = currentData.split(" ");
- Location location = new Location("dummyGPS");
- location.setLatitude(Double.parseDouble(latLongitude[0]));
- location.setLongitude(Double.parseDouble(latLongitude[1]));
- earthquakeEntry.setLocation(location);
- }
- });
-
- entry.getChild(GEORSS_NAMESPACE, kGeoRSSElevElementName).setEndTextElementListener(new EndTextElementListener() {
- @Override
- public void end(String currentData) {
-
-
- double evel;
-
- try {
- evel = Double.parseDouble(currentData);
- } catch (Exception e) {
-
- e.printStackTrace();
- evel = 0;
- }
- Log.v("Sax_Elev", String.valueOf(evel));
- earthquakeEntry.setElev(evel);
- }
- });
-
- entry.getChild(ATOM_NAMESPACE, kLinkElementName).setStartElementListener(new StartElementListener() {
- @Override
- public void start(Attributes attributes) {
-
-
- String webLink = attributes.getValue("href");
- earthquakeEntry.setLink(webLink);
- }
- });
-
- try {
- Xml.parse(inStream, Xml.Encoding.UTF_8, root.getContentHandler());
- } catch (Exception e) {
-
- e.printStackTrace();
- }
-
- return earthquakeEntryList;
- }
- }
开头处定义了解析需要的元素标签名称,因为getChild方法获取子元素时需要命名空间,因此还新定义了USGS的xml数据中包含的两个命名空间:
- static final String ATOM_NAMESPACE = "";
- static final String GEORSS_NAMESPACE = "";
在定义的用于解析xml数据的方法中
public ArrayList parse(InputStream inStream)
首先获取xml文档的根元素:
RootElement root = new RootElement(ATOM_NAMESPACE, kRootElementName);
有了根元素之后,就可以使用类似DOM的getChild方法获取具体的某个位置处的子元素,并且可以为具体的子元素注册事件处理器和在对应的回调函数中实现具体的处理逻辑。
从程序中我们可以看到,我们只为元素和包含的子元素注册了事件处理器,因此即使xml文档开始处有、和等同名的元素标签,但也不会进行处理。因此和上一部分的Demo相比,就不再需要设置标志变量用来判断,而且看起来也更加简单了。
虽然写法不同了,但是对具体标签的处理逻辑和上一部分Demo中是一样的,因为处理的xml文档内容没有变。
程序的最后调用android.util.Xml类的类方法parse直接进行解析,也更加方便了。
-
- try {
- Xml.parse(inStream, Xml.Encoding.UTF_8, root.getContentHandler());
- } catch (Exception e) {
-
- e.printStackTrace();
- }
其中Xml类parse方法的ContentHandler参数由根元素通过getContentHandler()方式获得。
最后添加AndroidXMLDemoSaxII.java文件中的内容,内容和上一个Demo工程AndroidXMLDemoSax中的AndroidXMLDemoSax.java基本一样,
- public class AndroidXMLDemoSaxII extends Activity {
-
-
- ListView list;
- ArrayAdapter adapter;
- ArrayList earthquakeEntryList;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
-
-
- InputStream earthquakeStream = readEarthquakeDataFromFile();
-
- AndroidSaxEarthquakeHandler androidSaxHandler = new AndroidSaxEarthquakeHandler();
- earthquakeEntryList = androidSaxHandler.parse(earthquakeStream);
-
- list = (ListView)this.findViewById(R.id.list);
- adapter = new ArrayAdapter(this, android.R.layout.simple_list_item_1, earthquakeEntryList);
- list.setAdapter(adapter);
- }
-
- private InputStream readEarthquakeDataFromFile()
- {
-
- InputStream inStream = null;
- try {
- inStream = this.getAssets().open("USGS_Earthquake_1M2_5.xml");
- } catch (IOException e) {
-
- e.printStackTrace();
- }
- return inStream;
- }
- private InputStream readEarthquakeDataFromInternet()
- {
-
- URL infoUrl = null;
- InputStream inStream = null;
- try {
- infoUrl = new URL("");
- URLConnection connection = infoUrl.openConnection();
- HttpURLConnection httpConnection = (HttpURLConnection)connection;
- int responseCode = httpConnection.getResponseCode();
- if(responseCode == HttpURLConnection.HTTP_OK)
- {
- inStream = httpConnection.getInputStream();
- }
- } catch (MalformedURLException e) {
-
- e.printStackTrace();
- } catch (IOException e) {
-
- e.printStackTrace();
- }
- return inStream;
- }
- }
只是把进行SAX解析的部分换成了如下方式:
-
- AndroidSaxEarthquakeHandler androidSaxHandler = new AndroidSaxEarthquakeHandler();
- earthquakeEntryList = androidSaxHandler.parse(earthquakeStream);
完成了,可以保存运行看下效果。
三.总结
Android平台提供了相当强大的XML解析支持,不仅包含了Java SDK中用来XML处理的API,而且Android SDK还提供了特有的用于SAX解析XML的包android.sax。使用这个包中的API可以更加方便解析,特别是当要解析的xml文档中在不同的层级位置处有多个相同名称的标签但需要分别进行不同处理时,同时也有更好的鲁棒性,减少解析时产生Bug的可能性。
以上我们介绍的都是SAX方式解析XML,而解析XML常用的还有DOM方式,这部分内容我们以后接着学习。
阅读(1304) | 评论(0) | 转发(0) |