解决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。
自己走了不少弯路。