您的位置:首页 > 移动开发 > Android开发

Android XML解析学习——方式比较 .

2013-08-30 17:42 369 查看
一.基础知识
经过这段时间的学习,我们已经了解了Android平台上用于解析XML的三种方式:SAX、DOM和Pull。并且在学习的过程中也介绍了这三种方式各自的特点及适合的使用场合,简单的来说,DOM方式最直观和容易理解,但是只适合XML文档较小的时候使用,而SAX方式更适合在Android系统中使用,因为相比DOM占用内存少,适合处理比较大的XML文档,最后的Pull方式使用场合和SAX类似,但是更适合需要提前结束XML文档解析的场合。
在这部分的学习中,我们将对以上三种方式解析XML的性能进行一下简单的比较,通过记录比较他们读取相同XML文档的时间来更好的理解他们的性能,从而使你在不同的场合更好的选择使用那一种XML的解析方式。

下面我们就用上面介绍的几种方式来实现解析XML形式的USGS地震数据的Demo例子以作比较。
二.实例开发
我们要完成的效果图如下图1所示:



图1 各种方式解析花费的时间比较



我们分别使用Java SAX,Android SAX,DOM和Pull方式解析相同的XML地震数据,并记录他们完成解析所花费的时间,结果如上图所示。
新建一个Android工程AndroidXMLDemoCompare。
添加进之前Demo工程AndroidXMLDemoSax中的EarthquakeEntry.java和SaxEarthquakeHandler.java文件,工程AndroidXMLDemoSaxII中的AndroidSaxEarthquakeHandler.java文件,工程AndroidXMLDemoDom中的DomEarthquakeHandler.java文件,和工程AndroidXMLDemoPull中的PullEarthquakeHandler.java文件。
如果需要从本地读取xml数据的话,同时在assets文件夹下添加保存为xml格式了的USGS地震数据USGS_Earthquake_1M2_5.xml和USGS_Earthquake_7M2_5.xml,如果需要联网读取的话,在manifest.xml文件中添加权限:




[xhtml]
view plaincopyprint?

<uses-permission android:name="android.permission.INTERNET" />

<uses-permission android:name="android.permission.INTERNET" />

并修改res/layout下的main.xml为:

[xhtml]
view plaincopyprint?

<?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"
>
<Button
android:id="@+id/javaSaxBtn"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Java SAX Parse"
>
</Button>
<Button
android:id="@+id/androidSaxBtn"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Android SAX Parse"
>
</Button>
<Button
android:id="@+id/domBtn"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="DOM Parse"
>
</Button>
<Button
android:id="@+id/pullBtn"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="PULL Parse"
>
</Button>
<TextView
android:id="@+id/javaSaxText"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Java SAX Parse Time:">
</TextView>
<TextView
android:id="@+id/androidSaxText"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Android SAX Parse Time:">
</TextView>
<TextView
android:id="@+id/domText"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="DOM Parse Time:">
</TextView>
<TextView
android:id="@+id/pullText"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="PULL Parse Time:">
</TextView>
</LinearLayout>

<?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" > <Button android:id="@+id/javaSaxBtn" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Java SAX Parse" > </Button> <Button android:id="@+id/androidSaxBtn" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Android SAX Parse" > </Button> <Button android:id="@+id/domBtn" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="DOM Parse" > </Button> <Button android:id="@+id/pullBtn" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="PULL Parse" > </Button> <TextView android:id="@+id/javaSaxText" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Java SAX Parse Time:"> </TextView> <TextView android:id="@+id/androidSaxText" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="Android SAX Parse Time:"> </TextView> <TextView android:id="@+id/domText" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="DOM Parse Time:"> </TextView> <TextView android:id="@+id/pullText" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="PULL Parse Time:"> </TextView> </LinearLayout>

主要定义了4个分别启动Java SAX、Android SAX、DOM和Pull方式解析的按钮Button,和4个显示解析所花费时间的TextView。
接着修改AndroidXMLDemoCompare.java文件的内容为:

[java]
view plaincopyprint?

public class AndroidXMLDemoCompare extends Activity {
/** Called when the activity is first created. */
//定义变量
Button javaSaxBtn, androidSaxBtn, domBtn, pullBtn;
TextView javaSaxText, androidSaxText, domText, pullText;
ArrayList<EarthquakeEntry> earthquakeEntryList;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);

//测试各个xml解析方法的速度
javaSaxBtn = (Button)findViewById(R.id.javaSaxBtn);
androidSaxBtn = (Button)findViewById(R.id.androidSaxBtn);
domBtn = (Button)findViewById(R.id.domBtn);
pullBtn = (Button)findViewById(R.id.pullBtn);
javaSaxText = (TextView)findViewById(R.id.javaSaxText);
androidSaxText = (TextView)findViewById(R.id.androidSaxText);
domText = (TextView)findViewById(R.id.domText);
pullText = (TextView)findViewById(R.id.pullText);
javaSaxBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
javaSaxBtn.setEnabled(false);
androidSaxBtn.setEnabled(false);
domBtn.setEnabled(false);
pullBtn.setEnabled(false);
//获取地震数据流
InputStream earthquakeStream = readEarthquakeDataFromFile();
String saxTime = "Java SAX Parse Time: 正在解析,请稍后...";
javaSaxText.setText(saxTime);
long beforeTime = System.currentTimeMillis();
//Java Sax方式进行xml解析
SAXParserFactory factory = SAXParserFactory.newInstance();
try {
SAXParser parser = factory.newSAXParser();
SaxEarthquakeHandler saxHandler = new SaxEarthquakeHandler();
parser.parse(earthquakeStream, saxHandler);
//获取解析后的列表数据
earthquakeEntryList = saxHandler.getEarthquakeEntryList();
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
//解析完毕
long afterTime = System.currentTimeMillis();
long parseTime = (afterTime - beforeTime);
saxTime = "Java SAX Parse Time: " + String.valueOf(parseTime) + "毫秒";
javaSaxText.setText(saxTime);
javaSaxBtn.setEnabled(true);
androidSaxBtn.setEnabled(true);
domBtn.setEnabled(true);
pullBtn.setEnabled(true);
}
});
androidSaxBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
javaSaxBtn.setEnabled(false);
androidSaxBtn.setEnabled(false);
domBtn.setEnabled(false);
pullBtn.setEnabled(false);
//获取地震数据流
InputStream earthquakeStream = readEarthquakeDataFromFile();
String saxTime = "Android SAX Parse Time: 正在解析,请稍后...";
androidSaxText.setText(saxTime);
long beforeTime = System.currentTimeMillis();
//Android Sax方式进行解析
AndroidSaxEarthquakeHandler androidSaxHandler = new AndroidSaxEarthquakeHandler();
earthquakeEntryList = androidSaxHandler.parse(earthquakeStream);
//解析完毕
long afterTime = System.currentTimeMillis();
long parseTime = (afterTime - beforeTime);
saxTime = "Android SAX Parse Time: " + String.valueOf(parseTime) + "毫秒";
androidSaxText.setText(saxTime);
javaSaxBtn.setEnabled(true);
androidSaxBtn.setEnabled(true);
domBtn.setEnabled(true);
pullBtn.setEnabled(true);
}
});
domBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
javaSaxBtn.setEnabled(false);
androidSaxBtn.setEnabled(false);
domBtn.setEnabled(false);
pullBtn.setEnabled(false);
//获取地震数据流
InputStream earthquakeStream = readEarthquakeDataFromFile();
String domTime = "DOM Parse Time: 正在解析,请稍后...";
domText.setText(domTime);
long beforeTime = System.currentTimeMillis();
//Dom方式进行xml解析
DomEarthquakeHandler domHandler = new DomEarthquakeHandler();
earthquakeEntryList = domHandler.parse(earthquakeStream);
//解析完毕
long afterTime = System.currentTimeMillis();
long parseTime = (afterTime - beforeTime);
domTime = "DOM Parse Time: " + String.valueOf(parseTime) + "毫秒";
domText.setText(domTime);
javaSaxBtn.setEnabled(true);
androidSaxBtn.setEnabled(true);
domBtn.setEnabled(true);
pullBtn.setEnabled(true);
}
});
pullBtn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
javaSaxBtn.setEnabled(false);
androidSaxBtn.setEnabled(false);
domBtn.setEnabled(false);
pullBtn.setEnabled(false);
//获取地震数据流
InputStream earthquakeStream = readEarthquakeDataFromFile();
String pullTime = "PULL Parse Time: 正在解析,请稍后...";
pullText.setText(pullTime);
long beforeTime = System.currentTimeMillis();
//Pull方式进行xml解析
PullEarthquakeHandler pullHandler = new PullEarthquakeHandler();
earthquakeEntryList = pullHandler.parse(earthquakeStream);
//解析完毕
long afterTime = System.currentTimeMillis();
long parseTime = (afterTime - beforeTime);
pullTime = "PULL Parse Time: " + String.valueOf(parseTime) + "毫秒";
pullText.setText(pullTime);
javaSaxBtn.setEnabled(true);
androidSaxBtn.setEnabled(true);
domBtn.setEnabled(true);
pullBtn.setEnabled(true);
}
});
}

private InputStream readEarthquakeDataFromFile()
{
//从本地获取地震数据
InputStream inStream = null;
try {
//1天内2.5级以上的地震数据,约20来条地震信息
// inStream = this.getAssets().open("USGS_Earthquake_1M2_5.xml");
//7天内2.5级以上的地震数据,约200来条地震信息
inStream = this.getAssets().open("USGS_Earthquake_7M2_5.xml");

} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return inStream;
}
private InputStream readEarthquakeDataFromInternet()
{
//从网络上获取实时地震数据
URL infoUrl = null;
InputStream inStream = null;
try {
//1天内2.5级以上的地震数据,约20来条地震信息
infoUrl = new URL("http://earthquake.usgs.gov/earthquakes/catalogs/1day-M2.5.xml");
//7天内2.5级以上的地震数据,约200来条地震信息
// infoUrl = new URL("http://earthquake.usgs.gov/earthquakes/catalogs/7day-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) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return inStream;
}
}

public class AndroidXMLDemoCompare extends Activity { /** Called when the activity is first created. */ //定义变量 Button javaSaxBtn, androidSaxBtn, domBtn, pullBtn; TextView javaSaxText, androidSaxText, domText, pullText; ArrayList<EarthquakeEntry> earthquakeEntryList; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); //测试各个xml解析方法的速度 javaSaxBtn = (Button)findViewById(R.id.javaSaxBtn); androidSaxBtn = (Button)findViewById(R.id.androidSaxBtn); domBtn = (Button)findViewById(R.id.domBtn); pullBtn = (Button)findViewById(R.id.pullBtn); javaSaxText = (TextView)findViewById(R.id.javaSaxText); androidSaxText = (TextView)findViewById(R.id.androidSaxText); domText = (TextView)findViewById(R.id.domText); pullText = (TextView)findViewById(R.id.pullText); javaSaxBtn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub javaSaxBtn.setEnabled(false); androidSaxBtn.setEnabled(false); domBtn.setEnabled(false); pullBtn.setEnabled(false); //获取地震数据流 InputStream earthquakeStream = readEarthquakeDataFromFile(); String saxTime = "Java SAX Parse Time: 正在解析,请稍后..."; javaSaxText.setText(saxTime); long beforeTime = System.currentTimeMillis(); //Java Sax方式进行xml解析 SAXParserFactory factory = SAXParserFactory.newInstance(); try { SAXParser parser = factory.newSAXParser(); SaxEarthquakeHandler saxHandler = new SaxEarthquakeHandler(); parser.parse(earthquakeStream, saxHandler); //获取解析后的列表数据 earthquakeEntryList = saxHandler.getEarthquakeEntryList(); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } //解析完毕 long afterTime = System.currentTimeMillis(); long parseTime = (afterTime - beforeTime); saxTime = "Java SAX Parse Time: " + String.valueOf(parseTime) + "毫秒"; javaSaxText.setText(saxTime); javaSaxBtn.setEnabled(true); androidSaxBtn.setEnabled(true); domBtn.setEnabled(true); pullBtn.setEnabled(true); } }); androidSaxBtn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub javaSaxBtn.setEnabled(false); androidSaxBtn.setEnabled(false); domBtn.setEnabled(false); pullBtn.setEnabled(false); //获取地震数据流 InputStream earthquakeStream = readEarthquakeDataFromFile(); String saxTime = "Android SAX Parse Time: 正在解析,请稍后..."; androidSaxText.setText(saxTime); long beforeTime = System.currentTimeMillis(); //Android Sax方式进行解析 AndroidSaxEarthquakeHandler androidSaxHandler = new AndroidSaxEarthquakeHandler(); earthquakeEntryList = androidSaxHandler.parse(earthquakeStream); //解析完毕 long afterTime = System.currentTimeMillis(); long parseTime = (afterTime - beforeTime); saxTime = "Android SAX Parse Time: " + String.valueOf(parseTime) + "毫秒"; androidSaxText.setText(saxTime); javaSaxBtn.setEnabled(true); androidSaxBtn.setEnabled(true); domBtn.setEnabled(true); pullBtn.setEnabled(true); } }); domBtn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub javaSaxBtn.setEnabled(false); androidSaxBtn.setEnabled(false); domBtn.setEnabled(false); pullBtn.setEnabled(false); //获取地震数据流 InputStream earthquakeStream = readEarthquakeDataFromFile(); String domTime = "DOM Parse Time: 正在解析,请稍后..."; domText.setText(domTime); long beforeTime = System.currentTimeMillis(); //Dom方式进行xml解析 DomEarthquakeHandler domHandler = new DomEarthquakeHandler(); earthquakeEntryList = domHandler.parse(earthquakeStream); //解析完毕 long afterTime = System.currentTimeMillis(); long parseTime = (afterTime - beforeTime); domTime = "DOM Parse Time: " + String.valueOf(parseTime) + "毫秒"; domText.setText(domTime); javaSaxBtn.setEnabled(true); androidSaxBtn.setEnabled(true); domBtn.setEnabled(true); pullBtn.setEnabled(true); } }); pullBtn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO Auto-generated method stub javaSaxBtn.setEnabled(false); androidSaxBtn.setEnabled(false); domBtn.setEnabled(false); pullBtn.setEnabled(false); //获取地震数据流 InputStream earthquakeStream = readEarthquakeDataFromFile(); String pullTime = "PULL Parse Time: 正在解析,请稍后..."; pullText.setText(pullTime); long beforeTime = System.currentTimeMillis(); //Pull方式进行xml解析 PullEarthquakeHandler pullHandler = new PullEarthquakeHandler(); earthquakeEntryList = pullHandler.parse(earthquakeStream); //解析完毕 long afterTime = System.currentTimeMillis(); long parseTime = (afterTime - beforeTime); pullTime = "PULL Parse Time: " + String.valueOf(parseTime) + "毫秒"; pullText.setText(pullTime); javaSaxBtn.setEnabled(true); androidSaxBtn.setEnabled(true); domBtn.setEnabled(true); pullBtn.setEnabled(true); } }); } private InputStream readEarthquakeDataFromFile() { //从本地获取地震数据 InputStream inStream = null; try { //1天内2.5级以上的地震数据,约20来条地震信息// inStream = this.getAssets().open("USGS_Earthquake_1M2_5.xml"); //7天内2.5级以上的地震数据,约200来条地震信息 inStream = this.getAssets().open("USGS_Earthquake_7M2_5.xml"); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } return inStream; } private InputStream readEarthquakeDataFromInternet() { //从网络上获取实时地震数据 URL infoUrl = null; InputStream inStream = null; try { //1天内2.5级以上的地震数据,约20来条地震信息 infoUrl = new URL("http://earthquake.usgs.gov/earthquakes/catalogs/1day-M2.5.xml"); //7天内2.5级以上的地震数据,约200来条地震信息// infoUrl = new URL("http://earthquake.usgs.gov/earthquakes/catalogs/7day-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) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } return inStream; }}

首先也是定义各个Button和TextView控件,接着为各个Button注册单击事件处理器,在单击事件处理的回调函数中,主要就是运行对应XML解析方式的解析过程,并且分别记录解析前和解析后的系统时间来计算解析所花费的时间,

[java]
view plaincopyprint?

long beforeTime = System.currentTimeMillis();
//Android Sax方式进行解析
AndroidSaxEarthquakeHandler androidSaxHandler = new AndroidSaxEarthquakeHandler();
earthquakeEntryList = androidSaxHandler.parse(earthquakeStream);
//解析完毕
long afterTime = System.currentTimeMillis();
long parseTime = (afterTime - beforeTime);

long beforeTime = System.currentTimeMillis(); //Android Sax方式进行解析 AndroidSaxEarthquakeHandler androidSaxHandler = new AndroidSaxEarthquakeHandler(); earthquakeEntryList = androidSaxHandler.parse(earthquakeStream); //解析完毕 long afterTime = System.currentTimeMillis(); long parseTime = (afterTime - beforeTime);

完成了,可以保存运行看下效果。






图2
解析时间比较

左图是解析1天内2.5级以上的地震数据,约20来条地震信息时各个解析方式所花费的时间,右图是解析7天内2.5级以上的地震数据,约180来条地震信息时各个解析方式所花费的时间。从上图我们可以看到Java
SAX、Android SAX和Pull方式花费的时间基本差不多,因为他们都是基于事件处理的方式,并且Java SAX和Android
SAX底层调用的都是相同的org.xml.sax包中XMLReader解析器。而DOM方式相对来说所花费的时间就会长点,在地震数据条数较少和较多时都比较长。因此不管是从内存的消耗角度或者解析使用的时间角度来考虑,在Android平台上的应用程序中都不太推荐使用DOM方式来解析XML数据。但是Java
SAX、Android SAX和Pull就看你喜欢使用哪个和你具体的使用场合了,性能上他们基本相同。

三.总结
在这部分内容中我们学习了各个解析方式的性能比较,即解析同一个XML文档时所花费时间的比较,并且从结果可以看出DOM方式的性能相对来说差点,而Java SAX、Android SAX和Pull方式的性能基本相同。
这样我们就比较全面的学习了Android平台上对XML文档进行解析的各种方式,但目前我们只是使用现成的XML文档来解析,实际上在使用过程中我们可能还会需要构造XML文档,比如可能会有需要向服务器发送XML,或者把一份数据以XML的形式保存在本地,这块内容就是写XML,我们以后接着学习
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: