好长时间没更新博客了,最近听取别人的博客写作经验,将开发中遇到的问题进行分享也是原创。这是去年的网约车项目中遇到的。
在处理多线程任务时,比如倒计时接单(通过子线程实现倒计时),新来订单15s后自动消失。而业务需求中又要允许用户在倒计时未结束时手动划掉消失,则此时需要停止倒计时线程。或者可能有时有多单依次叠加,此时最上面的一单消失时,也需要先停止上一单倒计时再自动开启下一单的倒计时。
如果是new Thread方式的interrupt()方法来停止线程,你打印时输出会发现倒计时中累加的变量并未停止累加。个中原因参看停止Java线程,小心interrupt()方法
万般无奈中想起AsyncTask不是自带线程停止方法么,即taskProgress.cancel(true)(taskProgress是TaskProgress继承AsyncTask后新new的对象),并且自带判断线程是否正在执行的方法taskProgress.getStatus() == AsyncTask.Status.RUNNING,果断解决问题。代码如下:
/*
* 新来订单时开始计时
*/
public static TaskProgress taskProgress;
public void startCountdown(final int mProgress, final int maxtime) {
if (maxtime > 0) {
roundProgressBar.setVisibility(View.VISIBLE);
if (taskProgress != null && taskProgress.getStatus() == AsyncTask.Status.RUNNING)
{
taskProgress.cancel(true); // 如果Task还在运行,则先取消它,然后执行关闭activity代码
taskProgress = null;
}
taskProgress = new TaskProgress(maxtime);
taskProgress.execute(0);
}
}
/*
* 开启倒计时,倒计时任务需要能手动停止,故而用AsyncTask,Thread无法在左右滑动卡片时停止线程
*/
public class TaskProgress extends AsyncTask<Integer, Integer, Integer> {// 继承AsyncTask
int maxtime;
int mProgress;
public TaskProgress(int maxtime) {
this.maxtime = maxtime;
}
@Override
protected Integer doInBackground(Integer... params) {// 处理后台执行的任务,在后台线程执行
int flag = 1;
if (isCancelled())
return null;
mProgress = params[0];
while (mProgress <= maxtime && !isCancelled()) {
try {
publishProgress(mProgress);//将会调用onProgressUpdate(Integer... progress)方法
mProgress += 1;
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
LogUtils.e("-----------------------", String.valueOf(mProgress));
LogUtils.e("-----------------------", String.valueOf(mProgress));
LogUtils.e("-----------------------", String.valueOf(mProgress));
}
return flag;
}
protected void onProgressUpdate(Integer... progress) {// 在调用publishProgress之后被调用,在ui线程执行
// roundProgressBar.setProgress(progress[0]);//更新进度条的进度
if (isCancelled()) {
return;
} else {
roundProgressBar.setProgress(progress[0], maxtime);
}
}
protected void onPostExecute(Integer result) {// 后台任务执行完之后被调用,在ui线程执行
if (result != null) {
//自动消失
MainActivity.getInstance().vanishOnDisappear();
}
}
protected void onPreExecute() {// 在doInBackground(Params...)之前被调用,在ui线程执行
roundProgressBar.setProgress(0, maxtime);// 进度条复位
}
protected void onCancelled() {// 在ui线程执行
// roundProgressBar.setProgress(0, maxtime);// 进度条复位
}
}
剩下的问题时,为什么这个异步任务的线程可以通过cancel方法停止呢?未完待续……..因为我还没看源码深究@_@。