Firefly开源社区

打印 上一主题 下一主题

【ZZP出品】Android解析XML文件

279

积分

10

威望

9

贡献

社区版主

Rank: 7Rank: 7Rank: 7

积分
279
QQ

【ZZP出品】Android解析XML文件

发表于 2014-10-21 09:54:29      浏览:9310 | 回复:1        打印      只看该作者   [复制链接] 楼主
本帖最后由 ZZP 于 2014-10-21 11:24 编辑
原文见Android解析XML文件

                        

一、在Android应用中的XML文件来源
1、本地xml文件
   本地XML文件可以放在应用根目录assets文件夹、res/xmlres/rawSDcard卡、应用的data目录等;
res/xml可直接通过getXml(intid)获取XML文档,返回一个解析器对象(XmlResourceParerXmlResourceParerXmlPullParser的子类),其它位置情况都可以获取XML文档,返回一个Inputstream对象,进行读取数据,获取方法分别如下:
a.res/xml目录下(推荐使用):
XmlResourceParser xmlParser = this.getResources().getXml(R.xml.XXX);  

b.
res/xmlres/raw目录下:
InputStream inputStream = this.getResources().openRawResource(R.xml.XXX);  

c.assets文件夹下(本人测试发现通过此方法获取的XML文档不能带有首行:<?xmlversion="1.0" encoding="utf-8"?>,否则解析报错,具体原因未查明,知道原因请回复交流)
InputStream inputStream = getResources().getAssets().open(fileName);  

d.
在应用指定目录下(SDcard,应用data目录等):
// path路径根据实际项目修改,此次获取SDcard根目录  
String path = Environment.getExternalStorageDirectory().toString();  
File xmlFlie = new File(path+fileName);  
InputStream inputStream = new FileInputStream(xmlFlie);  


2、通过url得到的xml文件
   很多时候需要解析xml文件都用于客户端与服务器之间的数据交互,比如解析google天气预报信息,或自己项目内定的一些XML数据结构,其中通过URL,使用DefaultHTTPClientget请求获取XML文件方法如下:
/**
* 读取urlxml资源 转成String
* @param url
* @return 返回 读取urlxml字符串
*/  
public String getStringByUrl(String url) {  
    String outputString = "";  
    // DefaultHttpClient  
    DefaultHttpClient httpclient = new DefaultHttpClient();  
    // HttpGet  
    HttpGet httpget = new HttpGet(url);  
    // ResponseHandler  
    ResponseHandler<String> responseHandler = new BasicResponseHandler();  
  
    try {  
        outputString = httpclient.execute(httpget, responseHandler);  
        outputString = new String(outputString.getBytes("ISO-8859-1"), "utf-8");    // 解决中文乱码  
  
        Log.i("HttpClientConnector", "连接成功");  
    } catch (Exception e) {  
        Log.i("HttpClientConnector", "连接失败");  
        e.printStackTrace();  
    }  
    httpclient.getConnectionManager().shutdown();  
    return outputString;  
}  



二、XML文件的解析方式
   能够运用在Android系统上解析XML文件的常用有三种方式:DOMSAXPULL,其中DOM解析XML是先把XML文件读进内存中,再通过接口获取数据,该方法使用相对小的XML文件,移动设备往往受硬件性能影响,如果XML文件比较大使用DOM解析往往效率跟不上;SAXPULL都是采用事件驱动方式来进行解析,在Android中的事件机制是基于回调函数。
   本例旨在考虑简单方便性,综合考虑选择了PULL解析,PULL解析器是一个开源项目,Android平台已经内置了PULL解析器,同时Android系统本身也是使用PULL解析器来解析各种XML文档。
   1、事件回调类型
   PULL解析XML文件时,回调XmlResourceParser内定义表示文档开头结束和节点开头结束的数值(事件回调类型),表示如下:
a.读取到XML文档开头(声明)返回:XmlPullParser.START_DOCUMENT0
b.读取到XML文档结束返回:XmlPullParser.END_DOCUMENT 1
c.读取到XML节点开始返回:XmlPullParser.START_TAG 2
d.读取到XML节点结束返回:XmlPullParser.END_TAG 3
e.读取到XML文本返回:XmlPullParser.TEXT 4
   
2XmlPullParser有几个主要方法(更多查阅AndroidAPIs):
a.XmlPullParser.getEventType(): Returns the type of the current event (START_TAG, END_TAG,TEXT, etc.) 【获取当前事件回调类型】
b.XmlPullParser.getName():ForSTART_TAG or END_TAG events, the (local) name of the current elementis returned when namespaces are enabled.【获取当前节点名字】
c.XmlPullParser.getAttributeValue(intindex):Returnsthe given attributes value.【根据id获取节点属性值】
d.XmlPullParser.getAttributeValue(String namespace, String name):Returnsthe attributes value identified by namespace URI and namespacelocalName.【根据name获取节点属性值】
e.XmlPullParser.netxText():Ifcurrent event is START_TAG then if next element is TEXT then elementcontent is returned or if next event is END_TAG then empty string isreturned, otherwise exception is thrown.【回调节点START_TAG时,通过此方法获取节点内容】

3、实际编码中如何使用
   在实际编码中,主要根据事件回调类型,结合被解析的XML结构进行解析提取数据,PULL解析XML文件的主要模式如下,更具体使用看本文提供的例子:
try {  
    //开始解析事件  
    int eventType = parser.getEventType();  
  
    //处理事件,不碰到文档结束就一直处理  
    while (eventType != XmlPullParser.END_DOCUMENT) {   
        //因为定义了一堆静态常量,所以这里可以用switch  
        switch (eventType) {  
            case XmlPullParser.START_DOCUMENT:  
                // 不做任何操作或初开始化数据  
                break;  
  
            case XmlPullParser.START_TAG:  
                // 解析XML节点数据  
                // 获取当前标签名字  
                String tagName = parser.getName();  
  
                if(tagName.equals("XXXTAGXXX")){  
  
                    // 通过getAttributeValue netxText解析节点的属性值和节点值  
  
                }  
                break;  
  
            case XmlPullParser.END_TAG:  
                // 单节点完成,可往集合里边添加新的数据  
                break;  
            case XmlPullParser.END_DOCUMENT:  
  
                break;  
        }  
  
        // 别忘了用next方法处理下一个事件,不然就会死循环  
        eventType = parser.next();  
    }  
} catch (XmlPullParserException e) {  
    e.printStackTrace();  
}catch (IOException e) {  
    e.printStackTrace();  
}  



三、运用例子
   本例主要是解析我国省市的XML文件,文件结构简单,本例采用资源数据地址:http://www.csw333.com/CityScene_I/getPlace.php,关于XML的结构基础之类的就不再累赘,实际项目中,根据XML文件的节点结果来进行变换,主要修改caseXmlPullParser.START_TAG部分代码。



   1.本例在获取XML文件的省份中,比较了获取三个不同位置的XML文件资源的读取方式;
   2.再获取url上的XML文件时,返回的字符串进行utf-8转码,避免中文乱码(本例采用的url资源数据需要转码);
   3.针对实际项目中,若xml文件过大,可以使用多线程进行解析,避免ANR,若解析时间较长,可以添加滚动提示;
   4.本例存在个Bug,根据例子使用的XML文件结构把直辖市也编进省份中,例子目的是解析XML文件,实际项目是需要避免;
   5.主要代码:
     a.各种获取XML文件资源方式
/**
* 同样删除首行,才能解析成功,
* @param fileName
* @return 返回xml文件的inputStream
*/      
public InputStream getInputStreamFromAssets(String fileName){  
    try {  
        InputStream inputStream = getResources().getAssets().open(fileName);  
        return inputStream;  
    } catch (IOException e) {  
        e.printStackTrace();  
    }  
    return null;  
}  
  
/**
* 读取XML文件,xml文件放到res/xml文件夹中,若XML为本地文件,则推荐该方法
*
* @param fileName
* @return : 读取到res/xml文件夹下的xml文件,返回XmlResourceParser对象(XmlPullParser的子类)
*/  
public XmlResourceParser getXMLFromResXml(String fileName){  
    XmlResourceParser xmlParser = null;  
    try {  
        //*/  
        //  xmlParser = this.getResources().getAssets().openXmlResourceParser("assets/"+fileName);        // 失败,找不到文件  
        xmlParser = this.getResources().getXml(R.xml.provinceandcity);  
        /*/  
        // xml文件在res目录下 也可以用此方法返回inputStream  
        InputStream inputStream = this.getResources().openRawResource(R.xml.provinceandcity);  
        /*/
        return xmlParser;
    } catch (Exception e) {
        e.printStackTrace();
    }
    return xmlParser;
}
/**
* 读取urlxml资源 转成String
* @param url
* @return 返回 读取urlxml字符串
*/  
public String getStringByUrl(String url) {  
    String outputString = "";  
    // DefaultHttpClient  
    DefaultHttpClient httpclient = new DefaultHttpClient();  
    // HttpGet  
    HttpGet httpget = new HttpGet(url);  
    // ResponseHandler  
    ResponseHandler<String> responseHandler = new BasicResponseHandler();  
  
    try {  
        outputString = httpclient.execute(httpget, responseHandler);  
        outputString = new String(outputString.getBytes("ISO-8859-1"), "utf-8");    // 解决中文乱码  
  
        Log.i("HttpClientConnector", "连接成功");  
    } catch (Exception e) {  
        Log.i("HttpClientConnector", "连接失败");  
        e.printStackTrace();  
    }  
    httpclient.getConnectionManager().shutdown();  
    return outputString;  
}  
  
/**
* 解析SDcard xml文件
* @param fileName
* @return 返回xml文件的inputStream
*/      
public InputStream getInputStreamFromSDcard(String fileName){  
    try {  
        // 路径根据实际项目修改  
        String path = Environment.getExternalStorageDirectory().toString() + "/test_xml/";  
  
        Log.v("", "path : " + path);  
  
        File xmlFlie = new File(path+fileName);  
  
        InputStream inputStream = new FileInputStream(xmlFlie);  
  
        return inputStream;  
    } catch (IOException e) {  
        e.printStackTrace();  
    }  
    return null;  
}  

     b.多线程解析:
/**
*  多线程加载网络端的xml,若xml文件过大也需要用该方式加载
*/  
Handler mHandler = new Handler();     
Runnable mRunnable = new Runnable() {  
    public void run() {  
        if(!isFinishParser){  
  
            mHandler.postDelayed(mRunnable, 1000);      
        }else{  
            showView.setText(provinceStr);  
            mHandler.removeCallbacks(mRunnable);  
        }  
    }  
};  
  
/**
* 比较耗时操作新建一个线程,避免UI线程ANR
*/  
public void parserWhitThread(){  
    new Thread(){  
        @Override  
        public void run() {                  
            provinceandcityStr = getStringByUrl(provinceAndCityUrl);  
            provinceArray = ProvincePullParse.Parse(provinceandcityStr);  
            for(Province pro : provinceArray){  
                provinceStr += pro.getProvinceId() + " : " +pro.getProvinceName()+"\n";  
            }  
            isFinishParser = true;  
        }  
    }.start();  
}  




     c.XML文件中解析出城市:
public static ArrayList<City> ParseXml(XmlPullParser parser){  
    ArrayList<City> CityArray = new ArrayList<City>();  
    City CityTemp = null;  
    int provinceId = 0;  
    int cityId;  
    String cityName;  
  
    try {  
        //开始解析事件  
        int eventType = parser.getEventType();  
  
        //处理事件,不碰到文档结束就一直处理  
        while (eventType != XmlPullParser.END_DOCUMENT) {  
            //因为定义了一堆静态常量,所以这里可以用switch  
            switch (eventType) {  
                case XmlPullParser.START_DOCUMENT:  
                    break;  
  
                case XmlPullParser.START_TAG:  
  
                    //给当前标签起个名字  
                    String tagName = parser.getName();  
                    //  Log.d("", "====XmlPullParser.START_TAG=== tagName: " + tagName);  
  
                    if(tagName.equals("province")){  
                        
                        provinceId = Integer.parseInt(parser.getAttributeValue(0));  
                    }else if(tagName.equals("item")){  
                        CityTemp = new City();  
                    }else if(tagName.equals("id")){  
                        cityId = Integer.parseInt(parser.nextText());                              
                        parser.next();  
                        cityName = parser.nextText();  
                          
                        Log.v("", "id getText: "+cityId);  
                        Log.v("", "name getText: "+cityName);                              
                        Log.e("", "=========================");  
                          
                        CityTemp.setProvinceId(provinceId);  
                        CityTemp.setCityId(cityId);  
                        CityTemp.setCityName(cityName);  
                          
                        CityArray.add(CityTemp);  
                    }  
                    break;  
  
                case XmlPullParser.END_TAG:  
                    break;  
                case XmlPullParser.END_DOCUMENT:  
                    break;  
            }  
  
            //别忘了用next方法处理下一个事件,忘了的结果就成死循环#_#  
            eventType = parser.next();  
        }  
    } catch (XmlPullParserException e) {  
        // TODO Auto-generated catch block  
        e.printStackTrace();  
    }catch (IOException e) {  
        // TODO Auto-generated catch block  
        e.printStackTrace();  
    }  
  
    return CityArray;  
}  

源码下载:

参考引用









回复

使用道具 举报

48

积分

0

威望

0

贡献

技术小白

积分
48
发表于 2016-11-23 21:25:22        只看该作者  沙发
谢谢分享啊
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

友情链接 : 爱板网 电子发烧友论坛 云汉电子社区 粤ICP备14022046号-2
快速回复 返回顶部 返回列表