注意:AsyncTask除了这里描述的内存泄漏之外,还有很多陷阱。因此,请谨慎使用此API,如果您不完全了解其含义,请完全避免使用它。有很多选择(线程,EventBus,RxAndroid等)。
一个常见的错误AsyncTask是捕获对主机Activity(或Fragment)的强烈引用:
class MyActivity extends Activity {
private AsyncTask<Void, Void, Void> myTask = new AsyncTask<Void, Void, Void>() {
//不要这样!内部类隐式保留指向它们的指针
// 父级,在这种情况下就是活动!
}
}这是一个问题,因为AsyncTask它很容易超过父级Activity,例如,如果在任务运行时发生配置更改。
正确的方法是使您的任务成为一个static类,该类不捕获父类,并持有对宿主的弱引用Activity:
class MyActivity extends Activity {
static class MyTask extends AsyncTask<Void, Void, Void> {
// 弱引用将仍然允许对活动进行垃圾收集
private final WeakReference<MyActivity> weakActivity;
MyTask(MyActivity myActivity) {
this.weakActivity= new WeakReference<>(myActivity);
}
@Override
public Void doInBackground(Void... params) {
// 在这里做异步的事情
}
@Override
public void onPostExecute(Void result) {
// 重新获得对该活动的强烈引用,并进行验证
// 它仍然存在并处于活动状态。
MyActivity activity = weakActivity.get();
if (activity == null
|| activity.isFinishing()
|| activity.isDestroyed()) {
// 活动已不再有效,请勿执行任何操作!
return;
}
// 该活动仍然有效,请在此处执行主线程操作
}
}
}