Android UI的操作并不是线程安全的,并且必须要在UI线程了里面执行(主线程)。
单线程模型原则:
1、不要阻塞UI线程
2、确保只有UI线程访问Android UI工具包
通常我们执行一些操作,为了不要阻塞UI线程,一些其他操作的常规做法是新开一个线程去执行,然后在其他线程里面是不能直接访问主UI成员,例如TextView.setText()等操作不允许访问。
android提供了几种其他线程访问主线程的方法:
1、Activity.runOnUiThread(Runnable) ;
2、View.post(Runnable) ;
3、View.postDelayed(Runnable) ;
4、Handler.
这些类和方法都能够访问主线程,但当实现一些复杂的操作时代码将变得很难理解。
为了解决这个问题,Android1.5开始提供了一个工具类:AsyncTask,它能使创建需要与用户界面交互的长时间任务变得简单,相对来说AsyncTask轻量级一点,适用于简单的异步处理,不需要借助于Handler。
AsyncTask是抽象类,定义了三个泛型类型(Params,Progress,Result)
Params:启动任务执行的输入参数,
Progress:后台任务的百分比
Result:后台任务最终返回结果
AsyncTask主要方法:
1、protected void onPreExecute() 方法:
在执行后用操作前被UI Thread调用,可以做一些准备工作。
2、protected void doInBackground(params) :
该方法运行在后台中,主要负责执行耗时的工作,可以调用onProgressUpdate()方法进行实时更新UI,为 抽象方法,子类必须实现。
3、protected void onProgressUpdate(values) 方法:
调用该方法后,UI线程将调用该方法实体进行UI的操作。
4、protected void onPostExecute(result) 方法:
后台线程doInBackground()方法执行完后,该方法被UI线程调用,返回结果。
这些方法不应该由应用程序调用,在任务启动时,方法会自动调用,启动方式:
new AsyncTask().execute(params,progressresult) ;
主要方法UML图:
执行流程图:
例子:(子线程更新进度条)
----------------------MainActivity.java----------------------
package com.conghua.activity;
import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.widget.SeekBar;
import android.widget.TextView;
public class MainActivity extends Activity {
SeekBar sbTest ;
TextView tvTest ;
private AsyncTask<Void,Integer,String> atTest = new AsyncTask<Void, Integer, String>(){
@Override
protected void onPreExecute() {
super.onPreExecute();
sbTest.setMax(100) ;
sbTest.setProgress(0) ;
}
@Override
protected String doInBackground(Void... params) {
for(int i=0; i<=100 ; i++){
onProgressUpdate(i) ;
try {
Thread.sleep(100) ;
} catch (InterruptedException e) {
e.printStackTrace();
}
}
return null;
}
@Override
protected void onProgressUpdate(Integer... values) {
super.onProgressUpdate(values);
sbTest.setProgress(values[0]) ;
}
@Override
protected void onPostExecute(String result) {
super.onPostExecute(result);
tvTest.setText("下载完成...") ;
}
} ;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
sbTest = (SeekBar) findViewById(R.id.sbTest) ;
tvTest = (TextView) findViewById(R.id.tvTest) ;
atTest.execute() ;
}
}
----------------------main.xml----------------------
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<SeekBar
android:id="@+id/sbTest"
android:layout_width="fill_parent"
android:layout_height="wrap_content"/>
<TextView
android:id="@+id/tvTest"
android:layout_width="fill_parent"
android:layout_height="wrap_content"/>
</LinearLayout>