package cn.com.saxtest;
import java.io.InputStream;
import java.util.ArrayList;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.widget.TextView;
/**
*
* @author chenzheng_java
* @description 使用sax解析方式解析xml
* @since 2010/03/04
*
*/
public class SaxParserActivity extends Activity {
private ArrayList<Beauty> beautyList = new ArrayList<Beauty>();
private Beauty beauty = null;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
SAXParserFactory factory = SAXParserFactory.newInstance();
try {
SAXParser parser = factory.newSAXParser();
InputStream inputStream = this.getClassLoader()
.getResourceAsStream("beauties.xml");
parser.parse(inputStream, new MyDefaultHandler());
String result = "";
for(Beauty beauty:beautyList){
result+="/n "+beauty.toString();
}
TextView textView = (TextView) findViewById(R.id.textView);
textView.setText(result);
} catch (Exception e) {
e.printStackTrace();
}
}
/**
*
* @author chenzheng_java
* @description 一定要注意,这里是使用的内部类哦。 有些朋友可能有疑问,不用内部类不可以,答案是:当然可以。
* 这里用内部类的目的其实就是为了提高一丁点效率而已。 如果你喜欢看源码,你就会发现,源码中大量的使用了内部类的。
*
*/
private class MyDefaultHandler extends DefaultHandler {
// 存储目前为止读取到的最后一个element的localname
private String currentElementName = "";
/**
* characters (char ch[], int start, int length)当解析xml中遇到文本内容时会执行。 ch
* 这个数组中存放的是整个xml文件的字符串的数组形式 start是当前解析的文本在整个xml字符串文件中的开始位置
* length是当前解析的文本内容的长度 由上面的介绍我们可以知道,我们可以通过new
* String(ch,start,length)方法来获取我们正解析的文本内容
*/
@Override
public void characters(char[] ch, int start, int length)
throws SAXException {
Log.i("currentElementName", currentElementName);
String textContent = new String(ch, start, length);
if(currentElementName.equals("name")&&textContent!=null&&!textContent.trim().equals("")){
Log.i("textContent name", textContent);
beauty.setName(textContent);
}
if(currentElementName.equals("age")&&textContent!=null&&!textContent.trim().equals("")){
Log.i("textContent age", textContent);
beauty.setAge(textContent);
}
}
/**
*解析到xml文档的末尾时触发
*/
@Override
public void endDocument() throws SAXException {
}
/**
* 解析到元素的末尾时触发
*/
@Override
public void endElement(String uri, String localName, String qName)
throws SAXException {
if(localName.equals("beauty")){
beautyList.add(beauty);
Log.i("beauty", beauty.toString());
}
}
/**
* 开始解析xml时触发
*/
@Override
public void startDocument() throws SAXException {
}
/**
* 解析到元素的开始处触发 startElement (String uri, String localName, String qName,
* Attributes attributes) uri:Namespace值,当用户没有明确指定以及当命名空间没有被使用的时候,为null
* localName:element的名称,或者通俗点叫标签的名称。如<name>中的name就是localName qName:
* 和localName的唯一其别是
* ,当标签有namespace时,该值返回的数据为全限定名称。例如<chen:name>中,localName为name
* ,qName为chen:name attributes:元素包含的属性对象。如果没有属性时,返回一个空的属性对象
*/
@Override
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
currentElementName = localName;
if(localName.equals("beauty")){
beauty = new Beauty();
}
}
}
/**
*
* @author chenzheng 这里使用内部类是为了效率考虑,内部类要比单独顶一个bean类更加的高效以及节约空间
*
*/
private class Beauty {
String name;
String age;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAge() {
return age;
}
public void setAge(String age) {
this.age = age;
}
@Override
public String toString() {
return "美女资料 [年龄=" + getAge() + ", 姓名=" + getName() + "]";
}
}