传送门 ☞ 轮子的专栏 ☞ 转载请注明 ☞ http://blog.csdn.net/leverage_1229html
最近单位来了一个Android4.1平台的360街景项目。在编写该项目demo的过程当中,为了省事,打算直接在UI线程中访问网络数据源并生成Bitmap以填充相应的视图。访问网络模块的封装采用了HttpClient的方式进行构建。编写完工后执行程序,发现视图显示的仍是本地的默认图样。在确认了网络权限已被开启的状况下,我开始怀疑是否是HttpClient封装的粒度过大,致使其适用范围受限的问题。因而干脆采用Java平台最底层的Socket套接字方式来实现网络访问,但是结果仍是同样的,仍旧没法获得网络数据。通过调试发现,在客户端发出请求以后,根本没法链接到服务端,也就没法解析后续的服务端的响应内容了。java
之前在Android2.3.3平台上研发怎么没有这个现象?难道是Android4.0的单线程模式的“禁令”较之以往更为严格了。为了使应用程序具备更好的交互性和更少的延迟时间,强势要求开发人员在UI主线程中只能执行与UI相关的工做(如:更新视图、与用户交互等),其余方面的工做一概禁止执行。为了验证这个相反,在stackoverflow.com检索了相关议题,从一位Google工程师的解答中基本获得了印证。也就是说,若是你非要在UI主线程中执行其余工做(如:访问网络、文件操做等),其实有这种编程强迫症的人在项目初期仍是不少的,笔者就是其中的一位。你须要为UI主线程所在的Activity设置线程策略,告知平台请赋予我在UI主线程中进行其余工做的权限。具体作法有以下:android
在你的Application、Activity或其它应用容器中添加以下代码:编程
public void onCreate() { if (DEVELOPER_MODE) { StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder() .detectDiskReads() // 捕捉读取磁盘 .detectDiskWrites() // 捕捉写入磁盘 .detectNetwork() // 捕捉网络访问 或使用detectAll() 火力全开 .penaltyLog() // 捕捉LogCat日志 .build()); StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder() .detectLeakedSqlLiteObjects() .detectLeakedClosableObjects() .penaltyLog() .penaltyDeath() .build()); } super.onCreate(); }
StrictMode是一个开发者工具类,从Android 2.3平台开始被引入。能够用于捕捉发生在应用程序UI主线程中耗时的IO操做、网络访问或方法调用;也能够用来改进应用程序,使得UI主线程在处理IO操做和访问网络时显得更平滑,避免其被阻塞,致使ANR警告。更多有关StrictMode的信息,请参见http://developer.android.com/reference/android/os/StrictMode.html。安全
这种很是规的作法,是在项目初期和开发模式下为了达到更高的效率,而采起一种提升生产效率作法。在产品交付和运维时,咱们仍是要严格遵照Android平台进程与线程安全管理机制。接下来是在实际开发中应该遵循的两个原则:网络
在UI主线程中,只处理与UI相关及用户交互的工做,耗时的工做一概交由后台工做线程去搭理。常见的耗时工做处理方式有:多线程
AsyncTask;app
Handler、MessageQueue、Looper;运维
ExecutorService(execute/submit)ide
在工做线程中,只作本身份内的事。决不干涉UI主线程的工做。在执行过程当中若是存在涉及到UI的操做(如:更新视图、重绘等),一概将其转交给UI主线程进行处理。常见的转交方式有:
Activity.runOnUiThread(new Runnable(){...});
View.post(new Runnable(){...});
View.postDelay(Runnable(){...},long)
最后,提供AsyncMultiThreadActivity演示Android多线程与UI交互的方式,仅供读者参考使用。
public class AsyncMultiThreadActivity extends Activity { private TextView txView; private Button button; /** Called when the activity is first created. */ @Override public void onCreate(Bundle savedInstanceState) { Log.i("RootyInfo", "oncreate"); super.onCreate(savedInstanceState); setContentView(R.layout.main); txView=(TextView)findViewById(R.id.textView1); button=(Button)findViewById(R.id.button1); button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { //建立一个用于展现前三种后台线程和UI线程交互的线程 new TestThread(MainActivity.this).start(); //建立一个用于展现AsyncTask实现交互的TestAsyncTask new TestAsyncTask().execute("Test"," AsyncTask"); } }); } class TestAsyncTask extends AsyncTask<String, Integer, String> { //TestAsyncTask被后台线程执行后,被UI线程被调用,通常用于初始化界面控件,如进度条 @Override protected void onPreExecute() { super.onPreExecute(); } //doInBackground执行完后由UI线程调用,用于更新界面操做 @Override protected void onPostExecute(String result) { txView.setText(result); super.onPostExecute(result); } //在PreExcute执行后被启动AysncTask的后台线程调用,将结果返回给UI线程 @Override protected String doInBackground(String... params) { StringBuffer sb=new StringBuffer(); for (String string : params) { sb.append(string); } return sb.toString(); } } //用于线程间通讯的Handler class TestHandler extends Handler { public TestHandler(Looper looper) { super(looper); } @Override public void handleMessage(Message msg) { System.out.println("123"); txView.setText((String)msg.getData().get("tag")); super.handleMessage(msg); } } //后台线程类 class TestThread extends Thread { Activity activity; public TestThread(Activity activity) { this.activity = activity; } @Override public void run() { // 演示Activity.runOnUIThread(Runnable)方法的实现 activity.runOnUiThread(new Runnable() { @Override public void run() { txView.setText("Test runOnUIThread"); } }); // 演示Activity.runOnUIThread(Runnable)方法的实现 txView.post(new Runnable() { @Override public void run() { txView.setText("Test View.post(Runnable)"); } }); // 演示Activity.runOnUIThread(Runnable)方法的实现 txView.postDelayed(new Runnable() { @Override public void run() { txView.setText("Test View.postDelay(Runnable,long)"); } }, 1000); // 演示Handler方法的实现 Message msg=new Message(); Bundle bundle=new Bundle(); bundle.putString("tag", "Test Handler"); msg.setData(bundle); new TestHandler(Looper.getMainLooper()).sendMessage(msg); super.run(); } } }