我有一个AsyncTask
,它在其doInBackground
中创建一个HttpClient并执行一个请求:
request = new HttpGet("http://myurl.com/");
try {
HttpClient client = new DefaultHttpClient();
httpResponse = client.execute(request);
} catch (IOException e) {
e.printStackTrace();
}
return httpResponse;
回到主线程,我尝试将响应的内容加载到位图中:
Bitmap bmp = BitmapFactory.decodeStream(httpResponse.getEntity().getContent());
这给我带来了NetworkOnMainThreadException,显然是因为我试图读取输入流。
为什么?我以为所有的网络都已经完成了,我只是在阅读回复的内容流。
堆栈跟踪为:
android.os.NetworkOnMainThreadException
at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1133)
at libcore.io.BlockGuardOs.recvfrom(BlockGuardOs.java:163)
at libcore.io.IoBridge.recvfrom(IoBridge.java:506)
at java.net.PlainSocketImpl.read(PlainSocketImpl.java:488)
at java.net.PlainSocketImpl.access$000(PlainSocketImpl.java:46)
at java.net.PlainSocketImpl$PlainSocketInputStream.read(PlainSocketImpl.java:240)
at org.apache.http.impl.io.AbstractSessionInputBuffer.fillBuffer(AbstractSessionInputBuffer.java:103)
at org.apache.http.impl.io.AbstractSessionInputBuffer.read(AbstractSessionInputBuffer.java:134)
at org.apache.http.impl.io.ContentLengthInputStream.read(ContentLengthInputStream.java:174)
at org.apache.http.impl.io.ContentLengthInputStream.read(ContentLengthInputStream.java:188)
at org.apache.http.conn.EofSensorInputStream.read(EofSensorInputStream.java:178)
at java.io.BufferedInputStream.fillbuf(BufferedInputStream.java:142)
at java.io.BufferedInputStream.read(BufferedInputStream.java:309)
at android.graphics.BitmapFactory.nativeDecodeStream(Native Method)
at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:530)
at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:603)
at myapp.onRestResponse(MyActivity.java:66)
at myapp.RestRequest.onPostExecute(RestRequest.java:145)
at myapp.RestRequest.onPostExecute(RestRequest.java:1)
at android.os.AsyncTask.finish(AsyncTask.java:631)
at android.os.AsyncTask.access$600(AsyncTask.java:177)
at android.os.AsyncTask$InternalHandler.handleMessage(AsyncTask.java:644)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:137)
at android.app.ActivityThread.main(ActivityThread.java:5103)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:525)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
at dalvik.system.NativeStart.main(Native Method)
11-30 00:00:41.976: D/skia(1642): ---- read threw an exception
11-30 00:00:41.976: D/skia(1642): --- decoder->decode returned false
11-30 00:02:38.836: D/AndroidRuntime(1739): Shutting down VM
11-30 00:02:38.836: W/dalvikvm(1739): threadid=1: thread exiting with uncaught exception (group=0x414c4700)
抛出此异常是因为您正在其主线程上执行位图操作。在AsyncTask:中运行代码
class Retreive extends AsyncTask<Void,Void,Void> {
private Exception exception;
protected Void doInBackground(String... urls) {
request = new HttpGet("http://myurl.com/");
try {
HttpClient client = new DefaultHttpClient();
httpResponse = client.execute(request);
} catch (IOException e) {
e.printStackTrace();
}
}
protected void onPostExecute(RSSFeed feed) {
// Write your bitmap code here
// TODO: check this.exception
// TODO: do something with the feed
}
}
不要忘记将此添加到AndroidManifest.xml文件:
这是一个更容易的方法,但你应该去Asnyc任务:-
StrictMode.ThreadPolicy policy = new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
从响应中读取流也请求使用网络。不要返回HttpResponse,而是像一样直接返回Bitmap
request = new HttpGet("http://myurl.com/");
try {
HttpClient client = new DefaultHttpClient();
httpResponse = client.execute(request);
Bitmap bmp = BitmapFactory.decodeStream(httpResponse.getEntity().getContent());
return bmp;
catch(Exception) {
e.printStackTrace();
return null;
}
解码Stream也会给UI线程带来负担。所以解码后台中的流