概念
當一個Android應用啓動之後,Android系統會爲這個應用程序創建一個主線程,該線程負責渲染圖像、分發事件、對界面進行輪詢監聽,也叫UI線程。
編程原則
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);
}
}