Android系统以不同寻常的方式处理多个应用程序的同时运行。来自于其它不同平台的开发者或许会对这样的运行机制感到很奇怪。而理解 Android多任务的运行,对于设计出可以良好运行的应用程序,以及与Android平台的其它部分进行无缝结合都具有重要意义。这篇文章说明了 Android的多任务方式设计上的成因,它对应用程序运行产生的影响,还有你可以怎样更好地利用Android的这一特性。
近期项目中,遇到一个问题,列表数据中的图片地址是一个需要下载JS再解析的字段,之前的图片下载是一个异步的过程,由一个队列处理。
public class ImageTaskExecutor {
/** 存放任务的链表,first-in last-out */
private LinkedList<ImageTask> mTaskQueue = null;
/** 执行任务的线程 */
private ThreadUnit mThreadUnit = null;
/** 执行任务的间隔时间 */
public static final long WAIT_PERIOD = 50L;
private volatile boolean paused;
private final Object signal = new Object();
/**
* 添加新任务
*
* @param task
* @return 是否添加成功
*/
public synchronized boolean addNewTask(final ImageTask task) {
if (mThreadUnit == null) {
mThreadUnit = new ThreadUnit();
mTaskQueue = new LinkedList<ImageTask>();
new Thread(mThreadUnit).start();
}
return mTaskQueue.offer(task);
}
class ThreadUnit implements Runnable {
public boolean isRunning = false;
private ImageTask task = null;
@Override
public void run() {
try {
isRunning = true;
while (isRunning) {
while (mTaskQueue != null && mTaskQueue.isEmpty()) {
try {
Thread.sleep(WAIT_PERIOD);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
synchronized (signal) {
while (paused) { // pause point
signal.wait();
}
}
if (mTaskQueue != null && !mTaskQueue.isEmpty()) {
task = mTaskQueue.removeFirst(); // 取出链表中的最后一个任务
if (task != null) {
task.execute();
}
}
} // end while
} catch (Exception e) {
e.toString();
}
} // end run
}
/**
* 中断任务的执行
*/
public void pauseTaskThread() {
setPaused();
}
private void setPaused() {
synchronized (signal) {
paused = true;
}
}
private void setUnpaused() {
synchronized (signal) {
paused = false;
signal.notify();
}
}
/**
* 恢复任务的执行
*/
public void resumeTaskThread(){
setUnpaused();
}
/**
* 终止任务的执行
*/
public void terminateTaskThread() {
if (mThreadUnit != null) {
mThreadUnit.isRunning = false;
}
if (mTaskQueue != null) {
mTaskQueue.clear();
}
mThreadUnit = null;
mTaskQueue = null;
}
}
现在列表中的图片信息需要解析,如果再开一个队列,页面直接卡得不动了。。。
然后,这时候就考滤整个下载JS然后再下载图片这个过程需要使用同步操作了
后来发现,这个过程操作会比较长,页面的开始出现第一项的图片闪跳
原来adapter里面的getView 方法,被调用的过程中,contentview里面的内容会被随机复用,然后就。。。
public void inflateTaobaoImage(final String jsonUrl, final View view,
final int error_bg_Id) {
if (jsonUrl == null || jsonUrl.equals("")) {
return;
}
String imgUrl = getImgUrl(jsonUrl);
if (!TextUtils.isEmpty(imgUrl)) {
final String originJsonUrl = (String) view.getTag(IMG_TAG);
if (TextUtils.equals(originJsonUrl, jsonUrl)) {
LogsPrinter.debugError(TAG, "inflateTaobaoImage in HashMap "
+ originJsonUrl + " " + view);
inflateImage(imgUrl, view, error_bg_Id);
}
} else {
mTaskExecutor.addNewTask(new ImageTask(jsonUrl) {
@Override
public void execute() {
String taobaoImgUrl = downloadUrlString(jsonUrl
+ "&callback=success_jsonpCallback");
final String imgUrl = getTaobaoImageUrl(taobaoImgUrl);
LogsPrinter.debugError("add map", imgUrl + " " + jsonUrl);
taobaoImgMap.add(imgUrl, jsonUrl);
final String originJsonUrl = (String) view.getTag(IMG_TAG);
if (TextUtils.equals(originJsonUrl, jsonUrl)) {
baseHandlers.post(new Runnable() {
@Override
public void run() {
inflateImage(imgUrl, view, error_bg_Id);
}
});
}
}
});
}
}
值得注意的一个问题是:
settag的值需要是一个固定的值。不然,有时候会出现加载多次的情况。。