Android HTTPUrlConnection 错误

Android HTTPUrlConnection errors

我正在为 Android 编写一个 Phonegap 应用程序,它使用我同时开发的一个插件。该应用程序使用 Rhino 运行 在 Java 脚本中执行一组数据分析操作。这样做的原因之一是让我能够在需要时静默更新数据分析规则,而无需强制更新整个应用程序版本。

Rhino使用的Java脚本代码由应用程序下载并存储到内部存储器。这有两种方式

两条路线都使用相同的底层 Java 代码来执行 JS 代码更新。第一个没有任何问题。但是第二个抛出异常。

相关代码

 private boolean doInit(CallbackContext cbc) 
 {
  Storage.setFilePath(this.context.getFilesDir());
  Storage.loadJSCode();
  ...
 }

上面的代码在 Phonegap 触发其 onDeviceReady 事件时调用 - i.e.when 应用程序启动。

Storage 是我在应用程序中处理文件 I/O 操作的通用 class。实际loadJSCode如下图

 public static String loadJSCode()
 {
  try
  {
   String rslt = "";
   HttpURLConnection conn = null;
   addr = "https://path/to/rhinocode.php";
   try
   {
    URL url = new URL(addr);
    conn = (HttpURLConnection)url.openConnection();
    conn.setRequestMethod("GET");
    conn.setRequestProperty("User-Agent", Statics.USER_AGENT);

    InputStream ips = conn.getInputStream();
    //THE EXCEPTION IS THROWN HERE
    int responseCode = conn.getResponseCode();
    if (200 != responseCode)
    {
     Feedback.logError("GET error: " + responseCode + " on " + addr); 
     return null;
    }

    BufferedReader bufr = new BufferedReader(new InputStreamReader(ips));
    String line;
    while ((line = bufr.readLine()) != null) rslt += line; 
    bufr.close();
   } finally{if (null != conn) conn.disconnect();} 
   return rslt;
  } catch(Exception e)
  {
   Feedback.logError("get fault " + Utils.stackTrace(e));   
   return null; 
  } 
 }
}

几个解释说明

触发对 loadJSCode 的违规调用的推送消息处理程序如下所示

 public class Receive extends BroadcastReceiver
 {
  @Override
  public void onReceive(Context context,Intent intent)
  {
   Bundle extras = intent.getExtras();
   if (extras != null) 
   {
    if (extras.getString("message") != null &&   
    extras.getString("message").length() != 0) 
    {
     try
     {   
      JSONObject jo = new JSONObject(extras.getString("message"));
      int code = jo.optInt("kind",0);
      if (1 == code) storage.loadJSCode();
     } catch(Exception e){Feedback.setError(e.getMessage());} 
    }
   }
  }
 }

导致此问题的完整堆栈跟踪如下所示

android.os.NetworkOnMainThread Exception 
at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictM ode.java:1 156) 
at com.android.org.conscrypt.OpenSSLSocketImpl.close(OpenSSLSock etImpl.javal 009) 
at com.android.okhttp.Connection.c1ose(Connection.java:175) 
at com.android.okhttp.internal.Util.c1oseQuietly(Util.java:110) 
at com.android.okhttp.internal.http.HttpEngine.release(HttpEngine.java: 447)
at com.android.okhttp.internal.http.HttpURLConnectionImpl.disconnect 
(HttpURLConnectionImpl.java:104) 
at  
com.android.okhttp.internal.http.HttpsURLConnectionlmpl.
disconnect(HttpsURLConnectionImpl java:124) 
at example.com.plugin.Storage.IoadJSCode( Storage.java:28) 
at example.com.plugin.Receive.onReceive (Receive.java:50) 
at android.app.ActivityThread.handle (Storage.java:28) 
at com.example.plugin.Receive.onReceive (Receive.java:50) 
at android.app.ActivityThread.handle Receiver(ActivityThread.java:2585) 
at android.app.ActivityThread.access 00(ActivityThread.java:151) 
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1409) 
at android.os.Handler.dispatchMessage(Handler.java:1 10) 
at android.os.Looper.loon(Looper.java)
00(ActivityThread.java 51) 
at android.app.ActivityThread$H.handleMessage(ActivityThread.java 409) 
at android.os.HandlerdispatchMessage(Handler.java:110) 
at android.os.Looper.loop(Looper java:193) 
at android.app.ActivityThread main( ActivityThread.java:5304) 
at java.lang.reflect.Method.invokeNative(Native Method) 
at java.lang.reflect.Method.invoke(Method.java:515) 
at com.android.internal.os.Zygotelnit $MethodAndArgsCaller.run(Zygote 
at android.os.Looper.loop(Looper.jav a:193) 
at android.app.ActivityThread.main( ActivityThread.java:5304) 
at java.lang.reflect.Method.invokeNative(Native Method) at    
java.lang.reflect.Method.invoke(Method.java:515) at 
com.android.internal.os.Zygotelnit 
$MethodAndArgsCaller.run(Zygot elnit.java:824) 
at com.android.internal.os.Zygotelnit .main(Zygotelnit.java:640) 
at dalvik.system.NativeStart.main

通过一些实验,我已经确定当 InputStream ips = conn.getInputStream(); 执行上述 loadJSCode 时会抛出异常。

为了完整起见,我应该提到 Receive BroadcastReceiver sub class 是通过 plugin.xml 文件中嵌入的以下内容触发的。

<receiver android:name="com.example.plugin.Receive" 
                                     android:exported="false">
 <intent-filter>
  <!-- Do not modify this -->
  <action android:name="pushy.me" />
 </intent-filter>
</receiver>

我正在使用 Phonegap CLI 6.3.3 和 Java 7.

我怀疑我在某种程度上 运行 违反了这里的线程规则,并最终从后台线程或类似的线程中做了一些事情。但是,我一点也不清楚我可能做错了什么导致了这种情况。我希望这里有人可以解释一下。

无法在主线程中执行网络操作。您应该在不同的线程中进行网络操作。您可以创建自己的线程或使用 android 框架提供的 AsyncTask。例如:

class AsyncTaskRunner extends AsyncTask<Void, Void, Void> {

  private String resp;

  @Override
  protected String doInBackground(String... params) {
//don't perform UI operation here, perform network operation
   resp = loadJSCode();
   return resp;
  }


  @Override
  protected void onPostExecute(String result) {
   // When network operation completed then this method is called and you 
  // can update UI from this method.
  }


  @Override
  protected void onPreExecute() {

  }


 }

现在你可以执行任务了:

new AsyncTaskRunner().execute();

更多详情https://developer.android.com/reference/android/os/AsyncTask.html