DocumentBuilder解析来自DefaultHttpClient的响应,但不解析来自更快的HttpURLConnection的响应

时间:2013-03-13 19:05:18

标签: android xml httpclient

我有以下类'MainActivity'它有2个方法downloadPageA和downloadPageB,一旦下载内容,它们都通过http连接检索内容,XMLfromString方法尝试将其解析为Document-Object。

问题: downloadPageA按预期工作,但downloadPageB在解析时崩溃。

类别:

package com.example.testhttp;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.StringReader;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.ProtocolException;
import java.net.URL;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.TransformerFactoryConfigurationError;  

import org.apache.http.HttpResponse;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.w3c.dom.Document;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException; 

import android.os.AsyncTask;
import android.os.Bundle;
import android.app.Activity;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            new Worker().execute("");
    }
    public static Document XMLfromString(String v) {
            Document doc = null;
            DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
            try {
                    DocumentBuilder db = dbf.newDocumentBuilder();
                    InputSource is = new InputSource();
                    is.setCharacterStream(new StringReader(v));
                    doc = db.parse(is);
            } catch (ParserConfigurationException e) {
                    e.printStackTrace();
            } catch (SAXException e) {
                    e.printStackTrace();
            } catch (IOException e) {
                    e.printStackTrace();
            }
            return doc;
    }

    public static String inputStreamToString(InputStream is) {
            String s = "";
            String line = "";
            BufferedReader rd = null;
            rd = new BufferedReader(new InputStreamReader(is));
            try {
                    while ((line = rd.readLine()) != null) {
                            s += line;
                    }
            } catch (IOException e) {
                    e.printStackTrace();
            }
            return s;
    }
    public static String downloadPageA(String url) {
            HttpClient httpclient = new DefaultHttpClient();
            HttpGet httpGet = new HttpGet(url);
            HttpResponse response = null;
            try {
                    response = httpclient.execute(httpGet);
                    return inputStreamToString(response.getEntity().getContent());
            } catch (ClientProtocolException e) {
            } catch (IOException e) {
            } catch (IllegalStateException e1) {
            } catch (TransformerFactoryConfigurationError e1) {
            }
            return null;
    }
    public static String downloadPageB(String url) {
            HttpURLConnection connection = null;
            URL serverAddress = null;
            try {
                    serverAddress = new URL(url);
                    connection = null;
                    connection = (HttpURLConnection) serverAddress.openConnection();
                    connection.setRequestMethod("GET");
                    connection.setReadTimeout(10000);
                    connection.connect();
                    return inputStreamToString(connection.getInputStream());
            } catch (MalformedURLException e) {
                    e.printStackTrace();
            } catch (ProtocolException e) {
                    e.printStackTrace();
            } catch (IOException e) {
                    e.printStackTrace();
            } finally {
                    connection.disconnect();
                    connection = null;
            }
            return "";
    }
    private class Worker extends AsyncTask<String, Void, String> {
            @Override
            protected String doInBackground(String... str) {
                    @SuppressWarnings("unused")
                    Document domA = XMLfromString(downloadPageA("http://en.wikipedia.com/wiki/Mario"));
                    @SuppressWarnings("unused")
                    Document domB = XMLfromString(downloadPageB("http://en.wikipedia.com/wiki/Mario"));

                    return "";
            }

            @Override
            protected void onPostExecute(String result) {
            }
            @Override
            protected void onPreExecute() {
            }
            @Override
            protected void onProgressUpdate(Void... values) {
            }
    }
    }

堆栈跟踪:

03-13 19:39:06.352: W/System.err(28733):    org.xml.sax.SAXParseException: Unexpected <! (position:START_DOCUMENT null@1:1 in java.io.StringReader@4166bfd0) 
03-13 19:39:06.352: W/System.err(28733):    at org.apache.harmony.xml.parsers.DocumentBuilderImpl.parse(DocumentBuilderImpl.java:146)
03-13 19:39:06.352: W/System.err(28733):    at com.example.testhttp.MainActivity.XMLfromString(MainActivity.java:64)
03-13 19:39:06.352: W/System.err(28733):    at com.example.testhttp.MainActivity$Worker.doInBackground(MainActivity.java:119)
03-13 19:39:06.352: W/System.err(28733):    at com.example.testhttp.MainActivity$Worker.doInBackground(MainActivity.java:1)
03-13 19:39:06.352: W/System.err(28733):    at android.os.AsyncTask$2.call(AsyncTask.java:287)
03-13 19:39:06.352: W/System.err(28733):    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
03-13 19:39:06.352: W/System.err(28733):    at java.util.concurrent.FutureTask.run(FutureTask.java:137)
03-13 19:39:06.352: W/System.err(28733):    at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:230)
03-13 19:39:06.352: W/System.err(28733):    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
03-13 19:39:06.352: W/System.err(28733):    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
03-13 19:39:06.362: W/System.err(28733):    at java.lang.Thread.run(Thread.java:856)

我想使用HttpUrlConnection,它更快,Android团队建议使用它而不是HttpClient。对这个错误的解释非常好!

马里奥

2 个答案:

答案 0 :(得分:1)

我明白了! downloadPageA downloadPageB 不会得到相同的结果,因为它们使用不同的用户代理字符串!

downloadPageB 我用过:

connection.setRequestProperty("User-Agent","Apache-HttpClient/UNAVAILABLE (java 1.4)");

在方法 downloadPageA 中将user-agent设置为等于HttpClient的用户代理。

答案 1 :(得分:-1)

您应该考虑为您的REST请求使用Android服务。 RoboSpice提供:

  • Android服务执行请求,比异步任务好得多(您是否曾尝试在请求期间轮换设备?)
  • 缓存
  • 如果使用SpringAndroid模块,则更容易解析...