解决android socket编程出现的Caused by: android.os.NetworkOnMainThreadException错
本文转自:http://blog.csdn.net/debutent/article/details/8708127
以前我的socket通信的代码运行在android2.3.5手机上一点问题没有,现在入手了小米2S(系统android4.1.1)后,发现出现如下问题,期间server端打开后异常关闭,在logcat中查看到了Caused by: android.os.NetworkOnMainThreadException错误,在http://stackoverflow.com/questions/13136539/caused-by-android-os-networkonmainthreadexception找到了答案,要在MainActivity.java的setContentView(R.layout.activity_main)后加入以下代码
if (android.os.Build.VERSION.SDK_INT > 9) { StrictMode.ThreadPolicy policy = new StrictMode.ThreadPolicy.Builder().permitAll().build(); StrictMode.setThreadPolicy(policy); }
并在文件头加入,import android.os.StrictMode
问题解决
对于什么是StrictMode,可以参考http://www.cnblogs.com/zelos/archive/2011/02/27/1966403.html
还有一篇文章解释的更清晰:
报android.os.NetworkOnMainThreadException异常,经过查文档,原来是4.0系统不允许主线程(UI线程)访问网络,因此导致了其异常。在4.0之后在主线程里面执行Http请求都会报这个错,也许是怕Http请求时间太长造成程序假死的情况吧。
一:在发起Http请求的Activity里面的onCreate函数里面添加如下代码:
//详见StrictMode文档
StrictMode.setThreadPolicy(newStrictMode.ThreadPolicy.Builder().detectDiskReads().detectDiskWrites().detectNetwork().penaltyLog().build()); StrictMode.setVmPolicy(newStrictMode.VmPolicy.Builder().detectLeakedSqlLiteObjects().detectLeakedClosableObjects().penaltyLog().penaltyDeath().build());
如果正在做的项目不是Android 4.0的是看不到StrictMode类的。
二:使用Thread、Runnable、Handler这三个类:
public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); this.setContentView(R.layout.share_mblog_view); newThread(runnable).start(); } Handler handler=newHandler(){ @Override public void handleMessage(Message msg) { super.handleMessage(msg); Bundle data=msg.getData(); String val= data.getString("value"); Log.i("mylog","请求结果为-->" +val); } } Runnable runnable=newRunnable(){ @Override public void run() { ////TODO: http request.// Message msg =newMessage(); Bundle data=newBundle(); data.putString("value","请求结果"); msg.setData(data); handler.sendMessage(msg); } }
开线程处理关于网络的事物,然后用handler发送消息来处理更新UI。
自己走了不少弯路。