java - Android HTTPUrlConnection 错误

标签 java android httpurlconnection phonegap-cli

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

Rhino 使用的 JavaScript 代码由应用程序下载并存储到内部存储中。这有两种方式发生

  • 应用启动时:当应用启动时,它会检查 JS 代码文件是否存在。如果不存在,它会从服务器获取它。
  • 响应推送消息:当需要时,我会向应用程序发送一条推送消息,该应用程序会尝试从服务器获取更新的 JS 代码。

两条路由都使用相同的底层 Java 代码来执行 JS 代码更新。第一个工作没有任何问题。然而,第二个抛出了异常。

相关代码

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

当 Phonegap 触发其 onDeviceReady 时,会调用上面的内容事件 - 即应用程序启动时。

Storage是我在应用程序中处理文件 I/O 操作的通用类。实际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; 
  } 
 }
}

一些解释说明

  • Feedback 是我用来处理涉及向用户显示内容的所有操作的类。
  • Utils 是我用来封装各种实用程序例程的类。在本例中,stackTrace 例程为我提供了导致问题的完整堆栈跟踪(作为字符串)。

触发对 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 $1800(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)
$1800(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子类通过 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

关于java - Android HTTPUrlConnection 错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39805141/

相关文章:

java - 如何在 Junit 4 中自定义 TestSuite 测试?

Android CSS 问题 : fantom white overlay (zoomable and touchable) in every browser very strange

android - 使用$ VARIABLE作为Jenkins中Gradle插件的构建任务

java - Google HTTP Java 客户端的 POST 请求错误

android - 在Android中上传大文件而不会出现内存不足错误

java - SMTP 服务器是否有可接受的语法? (以及如何解析它)

java - hibernate一对多将子对象与父关系分离

java - jsp 中捕获 nullpointerException 失败

android - 如何防止我的应用程序在手机旋转时重新启动?

java - HttpUrlConnection 到 HttpsUrlConnection,我可以只添加 's' 吗?