西木 2015-03-05
当一个Android应用启动之后,Android系统会为这个应用程序创建一个主线程,该线程负责渲染图像、分发事件、对界面进行轮询监听,也叫UI线程。
/* 运行效果:“按钮1”有持续一段时间的动画效果,点击“按钮2”模拟耗时操作阻塞UI线程 */ package com.example.ui_thread; import android.os.AsyncTask; import android.os.Bundle; import android.support.v7.app.ActionBarActivity; import android.util.Log; import android.view.Menu; import android.view.MenuItem; import android.view.View; import android.view.View.OnClickListener; import android.view.animation.TranslateAnimation; import android.widget.Button; import android.widget.TextView; public class MainActivity extends ActionBarActivity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); final TextView textView1=(TextView) findViewById(R.id.textView1); Button button1=(Button) findViewById(R.id.button1); Button button2=(Button) findViewById(R.id.button2); //为Button1添加Translate动画 TranslateAnimation animation=new TranslateAnimation(0, 150, 0, 0);//x坐标从0到150,y坐标不变 animation.setRepeatCount(30);//设置动画重复次数 animation.setDuration(2000);//单次执行动作的时间 button1.setAnimation(animation); //button2单击模拟ui阻塞 button2.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { //simulateChoke(textView1);//模拟耗时操作,界面会卡住不动,时间过久android系统会强制程序退出。 //solution1(textView1);//解决方案1:创建新的线程解决阻塞问题-创建一个新的线程,但是不能在非UI线程进行UI重绘操作,报异常。 //solution2(textView1,v);//解决方案2:创建新的线程解决阻塞问题-使用post方法,代码可读性差,维护性查。 solution3(textView1,v);//解决方案3 } }); } //耗时操作 private String loadingTextValue(int i1,int i2){ String result=i1+"--"+i2+"--"+System.currentTimeMillis(); try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); Log.e("event", e.getLocalizedMessage()); } return result; } //模拟耗时操作: private void simulateChoke(final TextView textView1) { Log.d("event", "betin -- 模拟阻塞"); String result=loadingTextValue(1,2);//耗时操作 textView1.setText(result); Log.d("event", "end -- 模拟阻塞"); } //解决方案1:创建新的线程解决阻塞问题: private void solution1(final TextView textView1) { //新起一个线程处理耗时操作 new Thread(new Runnable(){ @Override public void run(){ Log.d("event", "betin -- 解决阻塞方式1:new Thread"); String result=loadingTextValue(1,2); textView1.setText(result); Log.d("event", "end -- 解决阻塞方式1:new Thread"); } }).start(); } //解决方案2:创建新的线程解决阻塞问题: /*注意日志打印顺序: 03-05 03:49:45.543: D/event(588): begin -- 解决阻塞方式2:post方法 03-05 03:49:45.543: D/event(588): end -- 解决阻塞方式2:post方法 03-05 03:49:50.554: D/event(588): 1--2--1425527385553 03-05 03:49:50.554: D/event(588): begin -- v.post 03-05 03:49:50.554: D/event(588): end -- v.post 03-05 03:49:50.614: D/event(588): begin -- exec 03-05 03:49:50.614: D/event(588): end -- exec */ private void solution2(final TextView textView1, final View v) { Log.d("event", "begin -- 解决阻塞方式2:post方法"); new Thread(new Runnable(){ @Override public void run(){ //1.处理耗时操作 final String result=loadingTextValue(1,2); Log.d("event", result); //2.使用view.post方法解决在非UI线程中进行UI重绘的问题 Log.d("event", "begin -- v.post"); v.post(new Runnable(){ @Override public void run(){ Log.d("event", "begin -- exec "); textView1.setText(result); Log.d("event", "end -- exec "); } }); Log.d("event", "end -- v.post"); } }).start(); //但是该方式过于繁琐,需要自己维护Thread,可读性查、维护性查,见解决方案3 Log.d("event", "end -- 解决阻塞方式2:post方法"); } //解决方案3:继承android提供的AsyncTask工具类 /*注意日志打印顺序: 03-05 03:53:07.234: D/event(640): begin -- 解决阻塞方式3:AsyncTask 03-05 03:53:07.254: D/event(640): begin -- doInBackground 03-05 03:53:07.254: D/event(640): end -- 解决阻塞方式3:AsyncTask 03-05 03:53:12.253: D/event(640): end -- doInBackground 03-05 03:53:12.293: D/event(640): begin -- onPostExecute 03-05 03:53:12.303: D/event(640): end -- onPostExecute */ private void solution3(final TextView textView1, View v) { Log.d("event", "begin -- 解决阻塞方式3:AsyncTask"); new LoadingTestTask().execute(1,2); Log.d("event", "end -- 解决阻塞方式3:AsyncTask"); } private class LoadingTestTask extends AsyncTask<Integer, Void, String> { @Override protected String doInBackground(Integer... params) { Log.d("event", "begin -- doInBackground"); String result=loadingTextValue(params[0],params[1]);//耗时操作 Log.d("event", "end -- doInBackground"); return result; } @Override protected void onPostExecute(String result){ Log.d("event", "begin -- onPostExecute"); TextView textView1=(TextView) findViewById(R.id.textView1); textView1.setText(result); Log.d("event", "end -- onPostExecute"); } } @Override public boolean onCreateOptionsMenu(Menu menu) { // Inflate the menu; this adds items to the action bar if it is present. getMenuInflater().inflate(R.menu.main, menu); return true; } @Override public boolean onOptionsItemSelected(MenuItem item) { int id = item.getItemId(); if (id == R.id.action_settings) { return true; } return super.onOptionsItemSelected(item); } }