Android平台上HttpsURLConnectionImpl.getInputStream()错误

5

这是我的任务。给定一对纬度和经度,我想从Google Places API获取一个JSON。下面的函数reverseGeoCoding返回字符串JSON。

public String reverseGeoCoding(String lat, String lng) {
    latitude = lat;
    longitude = lng;
    String json = new String();
    String url = new String();

    URL googleAPI;
    try {
        url = URL + "location=" + latitude + "," + longitude + "&" + "radius=" + radius + "&" + "language=" + language + "&" + "sensor=false" + "&" + "key=" + APIKey;

        googleAPI = new URL(url);
        Log.d("TEST", googleAPI.toString());
        URLConnection gc = (URLConnection) googleAPI.openConnection();
        Log.d("TEST", gc.toString());

        InputStream is = gc.getInputStream();
        Log.d("TEST", "InputStream created");
        BufferedReader in = new BufferedReader(new InputStreamReader(is));
        Log.d("TEST", "reader created");
        String inputLine;
        while ((inputLine = in.readLine()) != null) {
            json = json + inputLine + "\n";
            Log.d("TEST", inputLine);
        }

        in.close();

    } catch (Exception e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }

    return json;
}

然而,getInputStream() 方法无法正常工作。我已经在 Web 浏览器上检查了 URL 请求,并且它可以正常工作(返回 JSON 文档)。

以下是错误信息。(为什么第 17 和 18 行会产生错误)

1. 12-11 00:46:02.862: W/System.err(12953): android.os.NetworkOnMainThreadException
2. 12-11 00:46:02.872: W/System.err(12953):     at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1084)
3. 12-11 00:46:02.872: W/System.err(12953):     at java.net.InetAddress.lookupHostByName(InetAddress.java:391)
4. 12-11 00:46:02.872: W/System.err(12953):     at java.net.InetAddress.getAllByNameImpl(InetAddress.java:242)
5. 12-11 00:46:02.872: W/System.err(12953):     at java.net.InetAddress.getAllByName(InetAddress.java:220)
6. 12-11 00:46:02.872: W/System.err(12953):     at libcore.net.http.HttpConnection.<init>(HttpConnection.java:71)
7. 12-11 00:46:02.872: W/System.err(12953):     at libcore.net.http.HttpConnection.<init>(HttpConnection.java:50)
8. 12-11 00:46:02.882: W/System.err(12953):     at libcore.net.http.HttpConnection$Address.connect(HttpConnection.java:351)
9. 12-11 00:46:02.882: W/System.err(12953):     at libcore.net.http.HttpConnectionPool.get(HttpConnectionPool.java:86)
10. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpConnection.connect(HttpConnection.java:128)
11. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpEngine.openSocketConnection(HttpEngine.java:308)
12. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpsURLConnectionImpl$HttpsEngine.makeSslConnection(HttpsURLConnectionImpl.java:460)
13. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpsURLConnectionImpl$HttpsEngine.connect(HttpsURLConnectionImpl.java:432)
14. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpEngine.sendSocketRequest(HttpEngine.java:282)
15. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpEngine.sendRequest(HttpEngine.java:232)
16. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpURLConnectionImpl.getResponse(HttpURLConnectionImpl.java:273)
17. 12-11 00:46:02.882: W/System.err(12953):    at libcore.net.http.HttpURLConnectionImpl.getInputStream(HttpURLConnectionImpl.java:168)
18. 12-11 00:46:02.894: W/System.err(12953):    at libcore.net.http.HttpsURLConnectionImpl.getInputStream(HttpsURLConnectionImpl.java:270)
19. 12-11 00:46:02.894: W/System.err(12953):    at locationpredict.POIList.reverseGeoCoding(POIList.java:55)
20. 12-11 00:46:02.894: W/System.err(12953):    at locationpredict.TestMain$1.onClick(TestMain.java:62)
3个回答

5

因为许多人在主线程上使用大量的网络操作,导致应用程序失去响应,因此添加了这个新功能来防止这种情况发生。
你应该使用AsyncTask或者service来完成这个操作。

        private class GetJsonTask extends AsyncTask<String, Void, String>{


            @Override
            protected String doInBackground(String... params) {
        try {
            latitude = params[0];
             longitude = params[1];
             String json = new String();
             String url = new String();
             URL googleAPI;


            url = URL + "location=" + latitude + "," + longitude + "&" + "radius=" + radius + "&" + "language=" + language + "&" + "sensor=false" + "&" + "key=" + APIKey;

            googleAPI = new URL(url);
            Log.d("TEST", googleAPI.toString());
            URLConnection gc = (URLConnection) googleAPI.openConnection();
            Log.d("TEST", gc.toString());

            InputStream is = gc.getInputStream();
            Log.d("TEST", "InputStream created");
            BufferedReader in = new BufferedReader(new InputStreamReader(is));
            Log.d("TEST", "reader created");
            String inputLine;
            while ((inputLine = in.readLine()) != null) {
                json = json + inputLine + "\n";
                Log.d("TEST", inputLine);
            }

            in.close();

        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
         return json;
    }

    @Override
        protected void onPostExecute(String result) {
            parseJson(result); // json should be stored in the result
        }
  }

private void parseJson(String json){
     JSONObject obj = new JSONObject(json);
....

}

这样调用它:
String[] data = new String[2];
data[0] = lat;
data[1] = long;
new GetJsonTask().execute(data);

或者你可以在调用方法之前,在你的onCreate()方法中添加以下代码来绕过它,但我不建议使用这种方法。只需将其用于测试即可。
   if( Build.VERSION.SDK_INT >= 9){
            StrictMode.ThreadPolicy policy = new StrictMode.ThreadPolicy.Builder().permitAll().build();
            StrictMode.setThreadPolicy(policy);
    }

还有一个问题,覆盖方法onPostExecute返回void。我如何从onPostExecute获取Google Places API的结果? - jaehui
我已经按照你的说明编写了解析doInBackground返回的JSON的代码,并将其放在了onPostExecute中。然而,调用异步方法(AsyncTask.execute(data))的主函数并不知道何时会调用onPostExecute方法,对吗? - jaehui
是的,这是另一个线程。您想如何处理来自JSON的数据? - Lazy Ninja
我想要将JSON解析并发送到另一个模块。 - jaehui
立即,ArrayList 是空的,因为 AsyncTask 将从网络加载。在 onPostExecute() 中,您将拥有数据,并且从那里您可以将它们发送到其他模块。 - Lazy Ninja
显示剩余2条评论

2

0

正如其他答案所建议的那样,使用异步任务。同时推荐使用gson将对象转换为/从json字符串!


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接