package com.czw.cancel;
import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.widget.ProgressBar;
import android.widget.Toast;
public class PbActivity extends Activity {
// 注意:
// 1.线程池是根据线程的顺序一个个的执行
// 2.安卓不可以对线程进行强制的停止
// 3.异步任务的cancel方法仅仅是把它标记为cancle的状态
// 4.这里可以把异步任务的cancle方法紧跟着Activity的生命周期(根据自己的需求定义,因为这里要测试界面关闭后线程是否还在运行)
// 5.AsyncTask.Status.RUNNING这个是AsyncTask的状态的获取方法
private ProgressBar pb;
private MyAsyncTask myAsyncTask;
private Toast toast;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_two);
pb = (ProgressBar) findViewById(R.id.pb);
toast = Toast.makeText(this, "当前进度值", 0);
// 启动异步:因为这里没有数据需要传递,所以execute方法中的参数为空
myAsyncTask = new MyAsyncTask();
myAsyncTask.execute();
}
// 当前界面失去焦点的回调方法
@Override
protected void onStop() {
// TODO Auto-generated method stub
super.onStop();
// 判断AsyncTask的状态(当前的AsyncTask是否不为空,并且处在运行的状态),否则就不用设置状态
if (myAsyncTask != null
&& myAsyncTask.getStatus() == AsyncTask.Status.RUNNING) {
myAsyncTask.cancel(true);
}
}
// 自定义的AsyncTask
public class MyAsyncTask extends AsyncTask<Void, Integer, Void> {
@Override
protected Void doInBackground(Void... params) {//注意这里不能进行UI的操作
// TODO Auto-generated method stub
for (int i = 0; i < 100; i++) {
if (myAsyncTask.isCancelled()) {// 判断是否为cancel的状态的时候就推出循环
break;
}
// 传递数据给onProgressUpdate方法
publishProgress(i);
try {
// 设置休眠300毫秒,这是模拟
new Thread().sleep(300);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
return null;
}
// UI更新调用的方法
@Override
protected void onProgressUpdate(Integer... values) {
// TODO Auto-generated method stub
super.onProgressUpdate(values);
//先判断是否继续更新进度条
if (myAsyncTask.isCancelled()) {// 判断是否为cancel的状态的时候就推出方法,因为是方法所以用return
return;
}
//获取数据并设置ProgressBar的进度,不过这里也是需要判断AsyncTask的状态
pb.setProgress(values[0]);
toast.setText("当前进度值"+values[0]);
toast.show();
}
}
}
常见问题:
1.AsycnTask的多次启动,会导致线程阻塞,也就是说,多次启动后,线程都会等待其他线程的执行完之后才再次执行新的线程(这是由于
AsycnTask的底层是通过线程池的实现的,当前线程没有启动完毕,之后的线程是不可以启动的
)
遇到的异常:
java.lang.RuntimeException: An error occured while executing doInBackground()
Caused by: android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
这是由于在
AsyncTask的
doInBackground方法中进行UI的操作,这个方法是不可以进行UI的操作的