Android HTTPUrlConnection 错误
Android HTTPUrlConnection errors
我正在为 Android 编写一个 Phonegap 应用程序,它使用我同时开发的一个插件。该应用程序使用 Rhino 运行 在 Java 脚本中执行一组数据分析操作。这样做的原因之一是让我能够在需要时静默更新数据分析规则,而无需强制更新整个应用程序版本。
Rhino使用的Java脚本代码由应用程序下载并存储到内部存储器。这有两种方式
- 应用程序启动时:当应用程序启动时,它会检查是否存在 JS 代码文件。如果不存在,则从服务器获取它。
- 响应推送消息:当需要时,我向应用程序发送推送消息,该应用程序通过尝试从服务器获取更新的 JS 代码来做出反应。
两条路线都使用相同的底层 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;
}
}
}
几个解释说明
- 反馈是 class 我用来处理所有涉及向用户显示内容的操作。
- Utils 是一个 class 我用来封装各种实用例程的工具。在这种情况下,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 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
我正在为 Android 编写一个 Phonegap 应用程序,它使用我同时开发的一个插件。该应用程序使用 Rhino 运行 在 Java 脚本中执行一组数据分析操作。这样做的原因之一是让我能够在需要时静默更新数据分析规则,而无需强制更新整个应用程序版本。
Rhino使用的Java脚本代码由应用程序下载并存储到内部存储器。这有两种方式
- 应用程序启动时:当应用程序启动时,它会检查是否存在 JS 代码文件。如果不存在,则从服务器获取它。
- 响应推送消息:当需要时,我向应用程序发送推送消息,该应用程序通过尝试从服务器获取更新的 JS 代码来做出反应。
两条路线都使用相同的底层 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;
}
}
}
几个解释说明
- 反馈是 class 我用来处理所有涉及向用户显示内容的操作。
- Utils 是一个 class 我用来封装各种实用例程的工具。在这种情况下,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 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