下面我们就用上面介绍的 Java SDK 中的SAX 方式来实现解析 XML 形式的地震数据的 Demo例子。
三. 实例开发
我们要解析的为美国地质调查局USGS 提供的地震数据, xml 数据地址为:
http://earthquake.usgs.gov/earthquakes/catalogs/1day-M2.5.xml
http://earthquake.usgs.gov/earthquakes/catalogs/7day-M2.5.xml
http://earthquake.usgs.gov/earthquakes/catalogs/7day-M5.xml
分别为1 天以内 2.5 级以上、 7 天内 2.5 级以上和 7 天内 5 级以上地震数据。
Xml数据的格式如下所示:
- <?xml version="1.0"?>
- <feed xmlns="http://www.w3.org/2005/Atom" xmlns:georss="http://www.georss.org/georss">
- <updated>2010-09-15T04:41:18Z</updated>
- <title>USGS M2.5+ Earthquakes</title>
- <subtitle>Real-time, worldwide earthquake list for the past day</subtitle>
- <link rel="self" href="http://earthquake.usgs.gov/earthquakes/catalogs/1day-M2.5.xml" mce_href="http://earthquake.usgs.gov/earthquakes/catalogs/1day-M2.5.xml"/>
- <link href="http://earthquake.usgs.gov/earthquakes/" mce_href="http://earthquake.usgs.gov/earthquakes/"/>
- <author><name>U.S. Geological Survey</name></author>
- <id>http://earthquake.usgs.gov/</id>
- <icon>/favicon.ico</icon>
- <entry>
- <id>urn:earthquake-usgs-gov:ak:10078833</id>
- <title>M 2.9, Southern Alaska</title>
- <updated>2010-09-15T04:14:03Z</updated>
- <link rel="alternate" type="text/html" href="http://earthquake.usgs.gov/earthquakes/recenteqsww/Quakes/ak10078833.php" mce_href="http://earthquake.usgs.gov/earthquakes/recenteqsww/Quakes/ak10078833.php"/>
- <summary type="html">
-
- </summary>
- <georss:point>59.9094 -153.1241</georss:point>
- <georss:elev>-98900</georss:elev>
- <category label="Age" term="Past hour"/>
- </entry>
- <entry>
-
- </entry>
- </feed>
下面我们就来完成用Java SAX 的方式解析以上 XML 形式的 USGS 地震数据的 Android 例子。
我们要完成的效果图如下图1 所示:
图1 ListView 列表显示的地震数据
解析完地震数据后用ListView 列表的方式显示每条地震的震级和地名信息。
新建一个Android 工程 AndroidXMLDemoSax 。
首先新建添加一个类EarthquakeEntry ,用来保存一条地震信息,类的内容为:
- public class EarthquakeEntry {
-
- private Date date;
- private Location location;
- private String place;
- private String link;
- private double magnitude;
- private double elev;
-
- public EarthquakeEntry()
- {
-
- }
- public EarthquakeEntry(Date _date, String _place, String _link, Location _location, double _magnitude, double _elev)
- {
- this.date = _date;
- this.location = _location;
- this.place = _place;
- this.link = _link;
- this.magnitude = _magnitude;
- this.elev = _elev;
- }
-
- public void setDate(Date _date)
- {
- this.date = _date;
- }
- public void setLocation(Location _location)
- {
- this.location = _location;
- }
- public void setPlace(String _place)
- {
- this.place = _place;
- }
- public void setLink(String _link)
- {
- this.link = _link;
- }
- public void setMagnitude(double _magnitude)
- {
- this.magnitude = _magnitude;
- }
- public void setElev(double _elev)
- {
- this.elev = _elev;
- }
-
- public Date getDate()
- {
- return this.date;
- }
- public Location getLocation()
- {
- return this.location;
- }
- public String getPlace()
- {
- return this.place;
- }
- public String getLink()
- {
- return this.link;
- }
- public double getMagnitude()
- {
- return this.magnitude;
- }
- public double getElev()
- {
- return this.elev;
- }
-
- @Override
- public String toString() {
- String earthquakeString = " M" + magnitude + " " + place;
- return earthquakeString;
- }
- }
比较简单,定义和一条地震内容对应的变量,并设置set 和 get 函数,并且重写 toString()函数在 ListView 输出时用。
接着新建添加一个类SaxEarthquakeHandler ,继承 DefaultHandler ,完成解析地震数据的具体逻辑实现,内容如下:
- public class SaxEarthquakeHandler extends DefaultHandler{
-
- private String kEntryElementName = "entry";
- private String kLinkElementName = "link";
- private String kTitleElementName = "title";
- private String kUpdatedElementName = "updated";
- private String kGeoRSSPointElementName = "point";
- private String kGeoRSSElevElementName = "elev";
-
- private ArrayList<EarthquakeEntry> earthquakeEntryList;
- private EarthquakeEntry earthquakeEntry;
- private StringBuilder currentDataBuilder;
- private Boolean startEntryElementFlag = false;
-
-
- public ArrayList<EarthquakeEntry> getEarthquakeEntryList()
- {
- return this.earthquakeEntryList;
- }
-
- @Override
- public void startDocument() throws SAXException {
- super.startDocument();
-
- earthquakeEntryList = new ArrayList<EarthquakeEntry>();
- currentDataBuilder = new StringBuilder();
- }
- @Override
- public void endDocument() throws SAXException {
-
- Log.v("Sax", "End");
- }
-
- @Override
- public void startElement(String uri, String localName, String qName,
- Attributes attributes) throws SAXException {
- super.startElement(uri, localName, qName, attributes);
-
- if(localName.equalsIgnoreCase(kEntryElementName))
- {
- earthquakeEntry = new EarthquakeEntry();
- startEntryElementFlag = true;
- }
- else if ((localName.equalsIgnoreCase(kLinkElementName))&&(startEntryElementFlag == true)) {
- String relAttribute = attributes.getValue("rel");
- if(relAttribute.equalsIgnoreCase("alternate"))
- {
- String webLink = attributes.getValue("href");
- earthquakeEntry.setLink(webLink);
- }
- }
- }
- @Override
- public void characters(char[] ch, int start, int length)
- throws SAXException {
- super.characters(ch, start, length);
- currentDataBuilder.append(ch, start, length);
- }
- @Override
- public void endElement(String uri, String localName, String qName)
- throws SAXException {
- super.endElement(uri, localName, qName);
- if(startEntryElementFlag == true)
- {
- String currentData = currentDataBuilder.toString();
- if (localName.equalsIgnoreCase(kTitleElementName)) {
-
- 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);
- }
- else if (localName.equalsIgnoreCase(kUpdatedElementName)) {
-
- 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);
- }
- else if (localName.equalsIgnoreCase(kGeoRSSPointElementName)) {
-
- String[] latLongitude = currentData.split(" ");
- Location location = new Location("dummyGPS");
- location.setLatitude(Double.parseDouble(latLongitude[0]));
- location.setLongitude(Double.parseDouble(latLongitude[1]));
- earthquakeEntry.setLocation(location);
- }
- else if (localName.equalsIgnoreCase(kGeoRSSElevElementName)) {
-
- double evel;
-
- try {
- evel = Double.parseDouble(currentData);
- } catch (Exception e) {
-
- e.printStackTrace();
- evel = 0;
- }
- Log.v("Sax_Elev", String.valueOf(evel));
- earthquakeEntry.setElev(evel);
- }
- else if(localName.equalsIgnoreCase(kEntryElementName))
- {
- earthquakeEntryList.add(earthquakeEntry);
- startEntryElementFlag = false;
- }
- currentDataBuilder.setLength(0);
- }
- }
- }
首先定义了 xml解析和保存解析结果等相关的一些变量,接着定义一个 get 函数
//获取解析的地震列表
public ArrayList<EarthquakeEntry> getEarthquakeEntryList()
{
return this . earthquakeEntryList ;
}
返回解析的地震列表数据。
然后就是具体的xml 解析回调函数的重写实现,因为继承了类 DefaultHandler ,包含了SAX 处理相关的 4 个 Handler 的所有回调函数的空实现,因此我们只需覆盖我们需要的回调函数。这是我们只重写了和文档内容处理相关的 ContentHandler 中 startDocument ,endDocument , startElement , characters ,和 endElement 这几个回调函数,在实际应用中你可能还需添加错误处理或者其他的回调函数,只需重写覆盖即可。
回调函数中具体的处理逻辑和你的XML 数据的内容有关,以上实现了解析 USGS 的地震数据的处理逻辑,并添加了注释,如果有兴趣你可以对照着 USGS 的 XML 数据格式来看,这就不具体的讲了。
和地震相关的存储类和SAX 处理回调函数都完成了,接下来我们就来调用SaxEarthquakeHandler 开始解析并显示数据。
先修改res/layout 下的 main.xml 为:
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/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>
添加了一个用于显示的ListView 控件。
接着添加AndroidXMLDemoSax.java 文件的内容。
先添加获取xml 数据源的方法:
- private InputStream readEarthquakeDataFromInternet()
- {
-
- URL infoUrl = null;
- InputStream inStream = null;
- try {
- infoUrl = new URL("http://earthquake.usgs.gov/earthquakes/catalogs/1day-M2.5.xml");
- 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;
- }
这是从 USGS的网站上读取 XML 数据并以 InputStream 的形式返回。因为需要用到联网功能,所以还得在 manifest.xml文件中添加联网权限:
< uses-permission android:name = "android.permission.INTERNET" />
这是联网获取XML 数据,也可以从本地读取 XML 数据,因为校园网会打不开 USGS 的网站,因此复制了一份 USGS 的地震数据以文件的形式保存在 assets 文件夹下,并使用如下函数读取:
- private InputStream readEarthquakeDataFromFile()
- {
-
- InputStream inStream = null;
- try {
- inStream = this.getAssets().open("USGS_Earthquake_1M2_5.xml");
- } catch (IOException e) {
-
- e.printStackTrace();
- }
- return inStream;
- }
有了XML 数据,就可以接下来进行解析了。
-
- InputStream earthquakeStream = readEarthquakeDataFromFile();
-
- SAXParserFactory factory = SAXParserFactory.newInstance();
- try {
- SAXParser parser = factory.newSAXParser();
- SaxEarthquakeHandler saxHandler = new SaxEarthquakeHandler();
- parser.parse(earthquakeStream, saxHandler);
-
- earthquakeEntryList = saxHandler.getEarthquakeEntryList();
- } catch (Exception e) {
-
- e.printStackTrace();
- }
最后添加定义相关变量,用把解析的数据用ListView 显示:
-
- ListView list;
- ArrayAdapter<EarthquakeEntry> adapter;
- ArrayList<EarthquakeEntry> earthquakeEntryList;
-
-
- list = (ListView)this.findViewById(R.id.list);
- adapter = new ArrayAdapter<EarthquakeEntry>(this, android.R.layout.simple_list_item_1, earthquakeEntryList);
- list.setAdapter(adapter);
完成了,可以保存运行看下效果。
以上使用的是javax.xml.parsers 包中的 SAXParser 来实现, SAXParser 包装了底层的 XMLReader ,实现起来更加方便。但是我们也可以使用 XMLReader 来实现解析,下面就看下使用具体的 XMLReader 的方式,实现代码如下所示:
-
- XMLReader xmlReader = null;
-
-
-
-
-
-
-
-
-
- try {
-
- System.setProperty("org.xml.sax.driver","org.xmlpull.v1.sax2.Driver");
- xmlReader = XMLReaderFactory.createXMLReader();
- } catch (SAXException e) {
-
- e.printStackTrace();
- }
- SaxEarthquakeHandler saxHandler = new SaxEarthquakeHandler();
- xmlReader.setContentHandler(saxHandler);
-
- InputSource inSource = new InputSource(earthquakeStream);
- try {
- xmlReader.parse(inSource);
- } catch (Exception e) {
-
- e.printStackTrace();
- }
-
- earthquakeEntryList = saxHandler.getEarthquakeEntryList();
其中获取XMLReader 的方式有两种,一种是先获取 SAXParser ,然后通过getXMLReader() 方法来获取,
xmlReader = SAXParserFactory.newInstance().newSAXParser().getXMLReader();
另一种是直接使用XMLReader 的工厂类 XMLReaderFactory 调用 createXMLReader()方法创建,
xmlReader = XMLReaderFactory. createXMLReader ();
但 不管是那种方式,只有获得了XMLReader 实例,接下来的操作都是一样的,先注册文档内容的事件处理器实例,
xmlReader.setContentHandler(saxHandler);
然后调用parse方法开始解析,
xmlReader.parse(inSource);
这样就用XMLReader进行了XML解析。
四. 总结
在这部分中我们首先学习了Android 上和 XML 解析相关的各个包的简单介绍,并且从有这么多个相关的包我们可以知道 Android 为 XML 的读写提供了相当大的支持。
然后具体学习了Android 上使用 SAX 方式解析 XML 的基本知识,使用 javax.xml.parsers 包中的 SAXParser 进行解析,及使用 org.xml.sax 包中的 XMLReader 进行解析两种方式分别的步骤,最后用解析 USGS 地震数据的 Demo 例子来实现介绍的内容。
这部分介绍的SAX 方式是属于原来 Java 就有的 XML 处理方式,同时, Android 平台为了使解析 XML 还能更加方便和更加健壮,提供了 android.sax 包来进行 SAX 进行 XML ,这部分内容我们以后我们继续接着学习。