0.  前言

在Android开发中经常会发生Activity的销毁重建,比如用户长时间接听一个电话后回到APP。在Android开发——Fragment知识整理(二)中我们提到了使用Fragment大量保存Activity销毁重建数据的方法,但是有一个问题是,在异步任务时旋转屏幕,如何处理异步任务呢?如果单纯的在Activity销毁之前关闭上一个异步任务,onPostExecute()中的关闭对话框就不会走了,会出现对话框无法关闭的现象;如果不关闭,可能会更新已经不存在的控件,造成错误,不仅如此最主要的是Activity的销毁会造成对话框dismiss空指针异常,因为与当前对话框绑定的FragmentManager已经是null。

因此我们的目标是在异步加载数据时旋转屏幕,不会对加载任务进行中断重启,并且对话框正常显示。

1.  继承Fragment并在其中声明引用

public class KeepDataFragment extends Fragment {
// 保存一个异步的任务
private MyAsyncTask data;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setRetainInstance(true);
}
public void setData(MyAsyncTask data) {
this.data = data;
}
public MyAsyncTask getData() {
return data;
}
}

这里我们创建KeepDataFragment并继承Fragment,并在其中声明需要保存的数据对象,这里是保存了一个异步的任务,然后提供getter和setter。最后一定要在onCreate调用setRetainInstance(true)。


2.  异步任务和进度条

public class MyAsyncTask extends AsyncTask<Void, Void, Void> {
private MainActivity activity;
private boolean isCompleted;
private LoadingDialog mLoadingDialog;
private List<String> items; public MyAsyncTask(MainActivity activity) {
this.activity = activity;
} @Override
protected void onPreExecute() {
mLoadingDialog = new LoadingDialog();
mLoadingDialog.show(activity.getFragmentManager(), "LOADING");
} @Override
protected Void doInBackground(Void... params) {
items = loadingData();
return null;
}
private List<String> loadingData() {
try {
Thread.sleep(5000);
} catch (InterruptedException e) {}
return new ArrayList<String>(Arrays.asList("东南大学", "信息科学与工程学院", "信息安全学科"));
} @Override
protected void onPostExecute(Void unused) {
isCompleted = true;
notifyActivityTaskCompleted();
if (mLoadingDialog != null)
mLoadingDialog.dismiss();
} public List<String> getItems() {
return items;
} public void setActivity(MainActivity activity) {
// 如果上一个Activity销毁,将与上一个Activity绑定的DialogFragment销毁
if (activity == null) {
mLoadingDialog.dismiss();
}
// 设置为当前的Activity
this.activity = activity;
// 开启一个与当前Activity绑定的等待框
if (activity != null && !isCompleted) {
mLoadingDialog = new LoadingDialog();
mLoadingDialog.show(activity.getFragmentManager(), "LOADING");
}
// 如果完成,通知Activity
if (isCompleted) {
notifyActivityTaskCompleted();
}
} private void notifyActivityTaskCompleted() {
if (null != activity) {
activity.onTaskCompleted();
}
}
}

这里使用AsyncTask进行异步任务,不熟悉AsyncTask的可以参考Android开发——AsyncTask的使用以及源码解析,任务开始时显示了一个FragmentDialog对话框,如果不熟悉可以参考Android开发——官方推荐使用DialogFragment替换AlertDialog,这里就不赘述了。任务下载中时,我们模拟了5秒耗时任务并返回了一个字符串List。下载任务结束时让进度框消失,并为Activity提供回调,因为这里持有了Activity的引用。这里我们也提供了setActivity方法,在Activity被销毁时在onSaveInstanceState()中设置setActivity(null)取消之前的对话框,同时也防止了内存泄漏;当Activity重建时在onCreate()中设置setActivity(this)传入新的Activity,从而再次显示一个加载框,这里需要注意的是Activity的销毁重建并不影响加载的数据,所有后台的数据一直继续在加载。

3.  MainActivity中的实现

public class MainActivity extends ListActivity {
private ListAdapter mAdapter;
private List<String> mDatas;
private KeepDataFragment dataFragment;
private MyAsyncTask mMyTask; @Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState); FragmentManager fm = getFragmentManager();
dataFragment = (KeepDataFragment) fm.findFragmentByTag("data");
if (dataFragment == null) {
dataFragment = new KeepDataFragment();
fm.beginTransaction().add(dataFragment, "data").commit();
}
mMyTask = dataFragment.getData();
if (mMyTask != null) {
//使AsyncTask持有Activity的引用
mMyTask.setActivity(this);
} else {
mMyTask = new MyAsyncTask(this);
dataFragment.setData(mMyTask);
mMyTask.execute();
}
} @Override
protected void onRestoreInstanceState(Bundle state) {
super.onRestoreInstanceState(state);
} @Override
protected void onSaveInstanceState(Bundle outState) {
mMyTask.setActivity(null);
super.onSaveInstanceState(outState);
} @Override
protected void onDestroy() {
super.onDestroy();
} public void onTaskCompleted() {
mDatas = mMyTask.getItems();
mAdapter = new ArrayAdapter<String>(MainActivity.this, android.R.layout.simple_list_item_1, mDatas);
setListAdapter(mAdapter);
}
}

在onCreate中,如果是第一次进入,则将Activity引用传给AsyncTask、开启任务mMyTask并把它交给KeepDataFragment来维护,正常情况下AsyncTask正常进行,完成后回调Activity中的onTaskCompleted()。注意要考虑不正常的情况,即加载过程中屏幕的旋转,Activity销毁时设置setActivity(null)取消之前的对话框,并在Activity重建时KeepDataFragment 实例因为未被销毁直接通过dataFragment.getData() 取出加载任务mTask并设置setActivity(this)从而再次显示一个新的加载框,直到任务完成正常进行Activity的回调显示数据方法。

看一下如下效果,加载数据的5秒钟内无论如何旋转屏幕都不会出现问题,这样就完成了进行异步任务时Activity的销毁重建不会发生中断并开启新的下载任务,而且对话框也会正常显示。

源码下载地址点这里

Android开发——异步任务中Activity销毁时的问题的更多相关文章

  1. Android开发之漫漫长途 Ⅱ——Activity的显示之Window和View(2)

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

  2. Android开发之漫漫长途 Ⅱ——Activity的显示之Window和View(1)

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

  3. Android开发之漫漫长途 Ⅲ——Activity的显示之Window和View(2)

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

  4. Android中Activity运行时屏幕方向与显示方式详解

    现在我们的手机一般都内置有方向感应器,手机屏幕会根据所处位置自动进行横竖屏切换(前提是未锁定屏幕方向).但有时我们的应用程序仅限在横屏或者竖屏状态下才可以运行,此时我们需要锁定该程序Activity运 ...

  5. Android开发艺术1之Activity的生命周期

    作为<Android开发艺术探索>这本书的第一篇博客,我就多说几句.本系列博客旨在对书中相关内容进行解读,简化,提供一个入门到提高的流程.不敢说书评,也不能说教程,只希望对有些人有帮助就好 ...

  6. Android开发学习之路--Activity之初体验

    环境也搭建好了,android系统也基本了解了,那么接下来就可以开始学习android开发了,相信这么学下去肯定可以把android开发学习好的,再加上时而再温故下linux下的知识,看看androi ...

  7. Android开发之漫漫长途 Ⅳ——Activity的显示之ViewRootImpl初探

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

  8. Android开发学习之路--Activity之生命周期

    其实这篇文章应该要在介绍Activity的时候写的,不过那个时候还不怎么熟悉Activity,还是在这里详细介绍下好了.还是参考下官方文档的图吧: 从上面的流程,我们可以看出首先就是打开APP,开始执 ...

  9. Android开发,Eclipse创建aidl接口时,出错

    Android开发中,当我们需要调用远程Service时,我们一般通过远程接口(RMI)来实现的,而Android的RMI需要AIDL(Android Interface Definition Lan ...

随机推荐

  1. Linux  释放Linux 系统预留的硬盘空间

    释放 Linux 系统预留的硬盘空间 by:授客 QQ:1033553122   大多数文件系统都会保留一部分空间作为紧急情况时用(比如硬盘空间满了),这样能保证有些关键应用(比如数据库)在硬盘满的时 ...

  2. 大数据【八】Flume部署

    如果说大数据中分布式收集日志用的是什么,你完全可以回答Flume!(面试小心问到哦) 首先说一个复制本服务器文件到目标服务器上,需要目标服务器的ip和密码: 命令: scp  filename   i ...

  3. 【疑难杂症04】EOFException异常详解

    最近线上的系统被检测出有错误日志,领导让我检查下问题,我就顺便了解了下这个异常. 了解一个类,当然是先去看他的API,EOFException的API如下: 通过这个API,我们可以得出以下信息: 这 ...

  4. EL表达式和标签

    1.什么是EL expression language 表达式语言 特点: 语言简单,使用方便 .${表达式}. 提供自动类型转换的功能 如果返回结果为null时 String -- ”” Numbe ...

  5. 看jQuery的这几天

    现在在做SPA时,有很多非常好用而且流行的前端框架,比如Vue,React,Angular等,jQuery似乎要逐渐退出前端的舞台了.不得不说,'write less,do more' 这句话吸引了我 ...

  6. Sql Server Tempdb原理-日志机制解析实践

    笔者曾经在面试DBA时的一句”tempdb为什么比其他数据库快?”使得95%以上的应试者都一脸茫然.Tempdb作为Sqlserver的重要特征,一直以来大家对它可能即熟悉又陌生.熟悉是我们时时刻刻都 ...

  7. python基础一数据类型之集合

    摘要: python基础一中介绍数据类型的时候有集合,所以这篇主要讲集合. 1,集合的定义 2,集合的功能 3,集合的方法 1,集合的定义 list1 = [1,4,5,7,3,6,7,9] set1 ...

  8. Oracle EBS AR 收款调整取值

    SELECT ct.trx_number ,adj.adjustment_number ,ad.amount_dr ,ad.amount_cr ,ad.source_table ,ad.source_ ...

  9. python基础学习18----面向对象简述

    这里就不再讲面向对象的相关概念知识或者与面向过程的比较了,直接进入类的学习 1.类的创建 class people: def __init__(self):#构造函数 pass sfencs=peop ...

  10. php 错误1

    Maximum execution time of 30 seconds exceeded 方法一,修改php.ini文件 max_execution_time = 30; Maximum execu ...