浅析网络数据解析

浅析网络数据解析

上一节我粗略的分享了一下使用HttpURLConnection和OKHttp的基本用法,接下来我的重点是分析一下获取网络数据解析。


1.1 解析XML格式数据

在网络上传输数据时候、最常用的格式有两种:XML和JSON,。下来我将一一分析这两种格式的解析。


1.1.1 Pull解析方式

解析XML格式的数据其实有很多种,我们比较常用的两种,Pull解析和SAX解析。

private void parseXMLWithPull(String xmlData) {
        try {
            XmlPullParserFactory factory = XmlPullParserFactory.newInstance();
            XmlPullParser xmlPullParser = factory.newPullParser();
            xmlPullParser.setInput(new StringReader(xmlData));
            int eventType = xmlPullParser.getEventType();
            String id = "";
            String name = "";
            String version = "";
            while (eventType != XmlPullParser.END_DOCUMENT) {
                String nodeName = xmlPullParser.getName();
                switch (eventType) {
                    // 开始解析某个结点
                    case XmlPullParser.START_TAG: {
                        if ("id".equals(nodeName)) {
                            id = xmlPullParser.nextText();
                        } else if ("name".equals(nodeName)) {
                            name = xmlPullParser.nextText();
                        } else if ("version".equals(nodeName)) {
                            version = xmlPullParser.nextText();
                        }
                        break;
                    }
                    // 完成解析某个结点
                    case XmlPullParser.END_TAG: {
                        if ("app".equals(nodeName)) {
                            Log.d("MainActivity", "id is " + id);
                            Log.d("MainActivity", "name is " + name);
                            Log.d("MainActivity", "version is " + version);
                        }
                        break;
                    }
                    default:
                        break;
                }
                eventType = xmlPullParser.next();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


在代码parseXMLWithPull()方法中,首先获取了XmlPullparserFactory的实例,并借助这个实例得到XmlPullParser对象,然后调用xmlPullParser的setInput()方法将服务器返回的XML数据设置进去就可以开始解析了,解析中通过getEventType()可以得到当前的解析事件,然后在一个while循环中不断地进行解析,如果当前的解析时间不等于xmlPullParser.End_DOCUMENT,说明解析工作还没有完成,调用next()方法后可以获取下一个解析事件。在while循环中,我们通过getName()方法得到当前节点的名字,如果发现节点名等于id、name或version,就调用nextText()方法来获取节点内具体的内容,每当解析完一个app节点后就将获取的内容打印出来.。按照上面代码示例应该会将XML数据中的制定内容成功的解析出来。


1.1.2 SAX解析

Pull解析方式虽然非常好用,但它并不是唯一的解析方式。SAX解析也是一种常用解析XML的解析方式,虽然它的用法比Pull解析复杂一些。

通常情况下我们都会新建一个类继承DefaultHandler,并重写父类的5个方法如下面代码示例:

/**
 * Description:
 * Copyright  : Copyright (c) 2016
 * Author     : yangfang
 */
 
public class ContentHandler extends DefaultHandler {

    private String nodeName;

    private StringBuilder id;

    private StringBuilder name;

    private StringBuilder version;

	//在开始XML解析的时候调用
    @Override
    public void startDocument() throws SAXException {
        id = new StringBuilder();
        name = new StringBuilder();
        version = new StringBuilder();
    }

	//在开始解析某个节点时候调用
    @Override
    public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
        // 记录当前结点名
        nodeName = localName;
    }

	//在获取节点内容中调用
    @Override
    public void characters(char[] ch, int start, int length) throws SAXException {
        // 根据当前的结点名判断将内容添加到哪一个StringBuilder对象中
        if ("id".equals(nodeName)) {
            id.append(ch, start, length);
        } else if ("name".equals(nodeName)) {
            name.append(ch, start, length);
        } else if ("version".equals(nodeName)) {
            version.append(ch, start, length);
        }
    }

	//在完成解析某个节点时调用
    @Override
    public void endElement(String uri, String localName, String qName) throws SAXException {
        if ("app".equals(localName)) {
            Log.d("ContentHandler", "id is " + id.toString().trim());
            Log.d("ContentHandler", "name is " + name.toString().trim());
            Log.d("ContentHandler", "version is " + version.toString().trim());
            // 最后要将StringBuilder清空掉
            id.setLength(0);
            name.setLength(0);
            version.setLength(0);
        }
    }

	//在完成整个XML解析的时候调用
    @Override
    public void endDocument() throws SAXException {
        super.endDocument();
    }

}
 

接下来修改MainAcitivity中的代码,如下面所示:

    private void parseXMLWithSAX(String xmlData) {
        try {
            SAXParserFactory factory = SAXParserFactory.newInstance();
            XMLReader xmlReader = factory.newSAXParser().getXMLReader();
            ContentHandler handler = new ContentHandler();
            // 将ContentHandler的实例设置到XMLReader中
            xmlReader.setContentHandler(handler);
            // 开始执行解析
            xmlReader.parse(new InputSource(new StringReader(xmlData)));
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


调用parseXMLWithSAX()方法来解析XML数据。parseXMLWithSAX()方法中先失创建一个SAXParserFactory的对象,然后再获取到XMLReader对象,接着ContentHandler的实例设置到XMLReader中,最后调用parse()方法开始执行解析就好了。

除了Pull解析和SAX解析之外,其实还有一种DOM解析方式,如果需要了解可以去查阅一下相关资料。


2.1 解析JSON格式数据

比起XML,JSON的主要优势在于它的体积更小,在网络上传输的时候可以更省流量。但是缺点在于它的语义性较差,看起来不如XML直观。


2.1.1使用JSONObject

类似地,解析JSON数据也有很多种方法,可以使用官方提供的JSONObject,也可以使用google的开源库GSON。另外,一些第三方的开源库如Jackson,FastJSON等也非常不错,此处只对这两种做介绍。

    private void parseJSONWithJSONObject(String jsonData) {
        try {
            JSONArray jsonArray = new JSONArray(jsonData);
            for (int i = 0; i < jsonArray.length(); i++) {
                JSONObject jsonObject = jsonArray.getJSONObject(i);
                String id = jsonObject.getString("id");
                String name = jsonObject.getString("name");
                String version = jsonObject.getString("version");
                Log.d("MainActivity", "id is " + id);
                Log.d("MainActivity", "name is " + name);
                Log.d("MainActivity", "version is " + version);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

2.1.2使用GSON


JSONObject来解析JSON数据已经非常简单了,Google提供的Gson开源库可以让解析JSON数据的工作简单。使用GSON库的依赖。编译app/build.gradle文件,在dependencies闭包中添加如下内容:

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    testCompile 'junit:junit:4.12'
    compile 'com.android.support:appcompat-v7:24.2.1'
    compile 'com.google.code.gson:gson:2.7'
}

具体实现如下代码

    private void parseJSONWithGSON(String jsonData) {
        Gson gson = new Gson();
        List<App> appList = gson.fromJson(jsonData, new TypeToken<List<App>>() {}.getType());
        for (App app : appList) {
            Log.d("MainActivity", "id is " + app.getId());
            Log.d("MainActivity", "name is " + app.getName());
            Log.d("MainActivity", "version is " + app.getVersion());
        }
    }


3.1网络编程的最佳实践

通常情况下我们都应该将这些通用的网络操作提到一个公共的类里,并提供一个静态的方法,如果需要发起网络请求的时候,只需简单地调用一下这个方法即可,代码如下:

首先需要定义一个借口,比如将它命名成HttpCallbackListener

public interface HttpCallbackListener {

    void onFinish(String response);

    void onError(Exception e);

}
在接口中定义了两个方法,onFinish()方法表示当服务器成功相应我们请求的时候调用,onError()表示当进行网络错做出现错误的时候调用。这两个方法都带有参数,onFinish()方法中的参数代表着服务器返回的数据,而onError()方法中的参数记录着错误的详细 

/**
 * Description:
 * Copyright  : Copyright (c) 2016
 * Author     : yangfang
 */
public class HttpUtil {

    public static void sendHttpRequest(final String address, final HttpCallbackListener listener) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                HttpURLConnection connection = null;
                try {
                    URL url = new URL(address);
                    connection = (HttpURLConnection) url.openConnection();
                    connection.setRequestMethod("GET");
                    connection.setConnectTimeout(8000);
                    connection.setReadTimeout(8000);
                    connection.setDoInput(true);
                    connection.setDoOutput(true);
                    InputStream in = connection.getInputStream();
                    BufferedReader reader = new BufferedReader(new InputStreamReader(in));
                    StringBuilder response = new StringBuilder();
                    String line;
                    while ((line = reader.readLine()) != null) {
                        response.append(line);
                    }
                    if (listener != null) {
                        // 回调onFinish()方法
                        listener.onFinish(response.toString());
                    }
                } catch (Exception e) {
                    if (listener != null) {
                        // 回调onError()方法
                        listener.onError(e);
                    }
                } finally {
                    if (connection != null) {
                        connection.disconnect();
                    }
                }
            }
        }).start();
    }

    public static void sendOkHttpRequest(final String address, final okhttp3.Callback callback) {
        OkHttpClient client = new OkHttpClient();
        Request request = new Request.Builder()
                .url(address)
                .build();
        client.newCall(request).enqueue(callback);
    }

}


使用网络请求的时候直接调用

HttpUtil.sendHttpRequest(address, new HttpCallbackListener(){
	
	@Override
	public void onFinish(String response){
		//在这里根据返回内容执行具体的逻辑
	}
	
	@Override
	public void onError(Exception e){
		//在这里对异常情况进行处理
		
	}
	
});

好了,简单的网络数据解析就分析到这里。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值