15_采用Pull解析器解析和生成XML內容
--------------------------------------
使用SAX或者DOM或者pull解析XML文件
------------------------------------------------------------
1.在Android平台上可以使用Simple API for XML(SAX) 、 Document Object Model(DOM)和
Android附帶的pull解析器解析XML文件。 pull解器是一個開源項目,既可以用於android
也可以用於javaEE,如過在javaEE中需要把其jar文件放入類路徑中,因為android已經集成了
Pull解析器,所以無需添加任何的jar包,android系統本身使用的各種xml文件,其內部也是采
Pull解析器進行解析的,Pull解析器的運行方式與SAX解析器相似,它提供了類似的事件,如
開始元素和結束元素事件,使用parser.net()可以進入下一個元素並觸發相應事件,跟SAX不
同的是,Pull解析器產生的事件是一個數字,而非方法,因此可以使用一個switch對感興趣的
事件進行處理,當元素開始解析時,調用parser.netText()方法,可以獲取下一個Text類型節
點的值
------------------------------------------------------
2.Pull解析器的源碼及文檔下載網址:http://xmlpull.org/xmlpull-website/impls.shtml
--------------------------------------------------------------------
<?xml version="1.0" encoding="UTF-8"?>//在xml文檔中start document
<persons>//根節點Node Type ?Element node/Textnode
<person id="23">//Element node類型節點
<name>李明</name>//Text node類型的節點
<age>30</age>
</person>
<person id="20">
<name>李向梅</name>
<age>25</age>
</person>
</persons>
//end documnet
--------------------------------------------
3.pull解析器的解析原理:
pull解析器會把xml文件讀到一個char[]數組中,當讀到一個<>號的時候,會判斷該<>中的
內容是否符合xml中的某個規則,符合就解析;當解析完一段之後,也就是一個<>之後就不會
往後解析,如果希望解析器往後解析需要調用api;
調用api後,會繼續往後解析,當再次解析的時候會觸發元素類型事件
-----------------------------------------------------------
4.java.lang.IllegalStateException: End of document.
at org.apache.harmony.xml.ExpatPullParser$EndDocumentEvent.getNext
(ExpatPullParser.java:697)
at org.apache.harmony.xml.ExpatPullParser$Document.dequeue
(ExpatPullParser.java:813)
at org.apache.harmony.xml.ExpatPullParser.next(ExpatPullParser.java:303)
at com.credream.service.PersonService.getPersons(PersonService.java:53)
at com.credream.test.PersonServiceTest.testPersons(PersonServiceTest.java:17)
at java.lang.reflect.Method.invokeNative(Native Method)
at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:169)
at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:154)
at android.test.InstrumentationTestRunner.onStart
(InstrumentationTestRunner.java:520)
at android.app.Instrumentation$InstrumentationThread.run
(Instrumentation.java:1447)
在解析的時候出現的錯誤
----------------------------------------------
5.SAX是一個解析速度快並且占用內存少的xml解析器,非常適合用於Android等移動設備。
SAX解析XML文件采用的是事件驅動,也就是說,它並不需要解析完整個文檔,在按內容順序
解析文檔的過程中,SAX會判斷當前讀到的字符是否合法XML語法中的某部分,如果符合就會
觸發事件。所謂事件,其實就是一些回調(callback)方法,這些方法(事件)定義在
ContentHandler接口。下面是一些ContentHandler接口常用的方法:
startDocument()
當遇到文檔的開頭的時候,調用這個方法,可以在其中做一些預處理的工作。
endDocument()
和上面的方法相對應,當文檔結束的時候,調用這個方法,可以在其中做一些善後的工作。
startElement(String namespaceURI, String localName, String qName, Attributes
atts)
當讀到一個開始標簽的時候,會觸發這個方法。namespaceURI就是命名空間,localName是不
帶命名空間前綴的標簽名,qName是帶命名空間前綴的標簽名。通過atts可以得到所有的屬性
名和相應的值。要注意的是SAX中一個重要的特點就是它的流式處理,當遇到一個標簽的時候
,它並不會紀錄下以前所碰到的標簽,也就是說,在startElement()方法中,所有你所知道
的信息,就是標簽的名字和屬性,至於標簽的嵌套結構,上層標簽的名字,是否有子元屬等
等其它與結構相關的信息,都是不得而知的,都需要你的程序來完成。這使得SAX在編程處理
上沒有DOM來得那麼方便。
endElement(String uri, String localName, String name)
這個方法和上面的方法相對應,在遇到結束標簽的時候,調用這個方法。
characters(char[] ch, int start, int length)
這個方法用來處理在XML文件中讀到的內容,第一個參數為文件的字符串內容,後面兩個參數
是讀到的字符串在這個數組中的起始位置和長度,使用new String(ch,start,length)就可以
獲取內容。
----------------------------------------------------
1.如何把應用中的數據存放到xml文件中:
StringBuilder xml=....先把xml組成一個字符串
xml.append(<xml>);然後將,xml寫到文件中
這種寫法不夠面向對象,生成文件的時候也容易出現語法問題
------------------------------------------------------------
2.用pull解析器讀取,以及寫入xml文件的代碼:
a.新建項目PullXml
/PullXml/src/com/credream/domain/Person.java
package com.credream.domain;
public class Person
{
private String name;
private Integer id;
private Integer age;
public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
public Integer getId()
{
return id;
}
public void setId(Integer id)
{
this.id = id;
}
public Integer getAge()
{
return age;
}
public void setAge(Integer age)
{
this.age = age;
}
@Override
public String toString()
{
return "Person [age=" + age + ", id=" + id + ", name=" + name +
"]";
}
public Person()
{
}
public Person(String name, Integer id, Integer age)
{
this.name = name;
this.id = id;
this.age = age;
}
}
--------------------------------------------------------------------
b.編寫xml操作的java文件
/PullXml/src/com/credream/service/PersonService.java
package com.credream.service;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.List;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlSerializer;
import android.util.Xml;
import com.credream.domain.Person;
public class PersonService
{
public static List<Person> getPersons(InputStream xml)throws Exception{
List<Person> persons=null;
Person person=null;
//得到Pull解析器的兩種方法//XmlPullParser
pullParser=XmlPullParserFactory.newInstance().newPullParser();
XmlPullParser pullParser=Xml.newPullParser();
pullParser.setInput(xml,"UTF-8");//為pull解析器設置要解析的xml數據
int event =pullParser.getEventType();
while(event!=XmlPullParser.END_DOCUMENT){//如果沒有到文檔結尾
switch (event){
case XmlPullParser.START_DOCUMENT://如果是開始節點
persons=new ArrayList<Person>();
break;
case XmlPullParser.START_TAG:
if("person".equals(pullParser.getName())){
int id=new Integer(pullParser.getAttributeValue(0));
person=new Person();
person.setId(id);
}
if("name".equals(pullParser.getName())){
String name=pullParser.nextText();//<name>李明</name>可以取得李明
person.setName(name);
}
if("age".equals(pullParser.getName())){
Integer age=new Integer(pullParser.nextText());//<name>李明</name>可以取
得李明
person.setAge(age);
}
break;
case XmlPullParser.END_TAG:
if("person".equals(pullParser.getName())){
persons.add(person);
person=null;
}
break;
}
event=pullParser.next();
}
return persons;
}
public static List<Person> getPersonsTest(InputStream xml) throws Exception{
List<Person> persons = null;
Person person = null;
XmlPullParser pullParser = Xml.newPullParser();
pullParser.setInput(xml, "UTF-8");//為Pull解析器設置要解析的XML數據
int event = pullParser.getEventType();
while(event != XmlPullParser.END_DOCUMENT){
switch (event) {
case XmlPullParser.START_DOCUMENT:
persons = new ArrayList<Person>();
break;
case XmlPullParser.START_TAG:
if("person".equals(pullParser.getName())){
int id = new Integer
(pullParser.getAttributeValue(0));
person = new Person();
person.setId(id);
}
if("name".equals(pullParser.getName())){
String name = pullParser.nextText();
person.setName(name);
}
if("age".equals(pullParser.getName())){
int age = new Integer(pullParser.nextText());
person.setAge(age);
}
break;
case XmlPullParser.END_TAG:
if("person".equals(pullParser.getName())){
persons.add(person);
person = null;
}
break;
}
event = pullParser.next();
}
return persons;
}
public static void save(List<Person> persons,OutputStream out)throws Exception{
XmlSerializer serializer=Xml.newSerializer();//序列化
serializer.setOutput(out, "UTF-8");
serializer.startDocument("UTF-8", true);//true指定是否可以單獨存在,而不依
賴其他xml文件
//<?xml version="1.0" encoding="UTF-8"?>//編碼是和xml中的編碼對應的
serializer.startTag(null, "persons");//第一個事命名空間
for(Person person:persons){
serializer.startTag(null, "person");
serializer.attribute(null, "id", person.getId().toString());
serializer.startTag(null,"name");
serializer.text(person.getName());
serializer.endTag(null, "name");
serializer.startTag(null, "age");
serializer.text(person.getAge().toString());
serializer.endTag(null, "age");
serializer.endTag(null, "person");
}
serializer.endTag(null, "persons");
serializer.endDocument();
out.flush();//如果用戶傳進的是個緩沖輸入流,那麼要flush下
out.close();
}
}
------------------------------------------------------------------
c.這個文件用來測試
/PullXml/src/com/credream/test/PersonServiceTest.java
package com.credream.test;
import java.io.File;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import com.credream.domain.Person;
import com.credream.service.PersonService;
import android.test.AndroidTestCase;
import android.util.Log;
public class PersonServiceTest extends AndroidTestCase
{
private static final String TAG="PersonServiceTest";
public void testPersons()throws Exception{
InputStream xml=this.getClass().getClassLoader().getResourceAsStream
("person.xml");
//List<Person> persons=PersonService.getPersonsTest(xml);
List<Person> persons=PersonService.getPersons(xml);
for(Person person :persons){
Log.i(TAG,person.toString() );
}
}
public void testSave()throws Exception{
List<Person> persons=new ArrayList<Person>();
persons.add(new Person("lidewei",42,22));
persons.add(new Person("lisha",42,24));
persons.add(new Person("xiaoyue",42,26));
persons.add(new Person("mimi",42,2));
//<包>/files
File xmlFile=new File(getContext().getFilesDir(),"credream.xml");
FileOutputStream outputStream =new FileOutputStream(xmlFile);
PersonService.save(persons, outputStream);
}
}
--------------------------------------------------------------
d.帶讀取的xml文件/PullXml/src/person.xml
<?xml version="1.0" encoding="UTF-8"?>
<persons>
<person id="23">
<name>李明</name>
<age>30</age>
</person>
<person id="20">
<name>李向梅</name>
<age>25</age>
</person>
</persons>
----------------------------------------------
e.清單文件:/PullXml/AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.credream.xml"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk android:minSdkVersion="8" />
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<activity
android:label="@string/app_name"
android:name=".PullXmlActivity" >
<intent-filter >www.2cto.com
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<uses-library android:name="android.test.runner" />
</application>
<instrumentation android:name="android.test.InstrumentationTestRunner"
android:targetPackage="com.credream.xml" android:label="Tests for My App" />
</manifest>
------------------------------------------------------------------