【问题标题】:What happens to an AsyncTask when the launching activity is stopped/destroyed while it is still running?当启动活动在它仍在运行时停止/销毁时,AsyncTask 会发生什么?
【发布时间】:2012-08-24 21:41:05
【问题描述】:

我见过几个几乎与我相同的问题,但我找不到一个完整的答案来满足我所有的疑问..所以我在这里..假设你有一个内部类扩展 @ 987654321@类是这样的:

public class MyActivity extends Activity {            
    private class DownloadImageTask extends AsyncTask<String, Void, Bitmap> { 
        protected Bitmap doInBackground(String... urls) {
            return DownloadImage(urls[0]); 
        }
        protected void onPostExecute(Bitmap result) {
            ImageView img = (ImageView) findViewById(R.id.img); 
            img.setImageBitmap(result);
        }  
    }

    @Override
    public void onCreate(Bundle savedInstanceState) { 
        super.onCreate(savedInstanceState); 
        setContentView(R.layout.main);
        new DownloadImageTask().execute("http://mysite.com/image.png")
    }
}

假设在DownloadImageTask仍在后台运行时activity被暂停或销毁(可能两种情况不同)。那么,在activity UI线程上运行的DownloadImageTask的方法可以被触发并且DownloadImageTask 可能会尝试使用暂停或销毁的 Activity 访问 Activity 的方法(它是一个内部类,因此它可以访问外部类的方法和实例变量),如下例中对 findViewByID 的调用..然后会发生什么?它会默默地失败吗?它会产生任何异常吗?是否会通知用户出现问题?

如果我们应该注意在调用 UI 上运行的方法时启动线程(在本例中为 Activity)仍然处于活动状态,我们如何在 AsyncTask 中实现这一点?

如果您发现这是一个重复的问题,我很抱歉,但也许这个问题更加清晰,有人可以更详细地回答

【问题讨论】:

  • 当您运行此代码时,它是否静默失败、产生任何异常或通知用户出现问题?
  • 我还没有运行这样的代码,我会在接下来的几天里运行它。在我的应用程序实现之前,我想知道它。一旦我可以运行测试,我会更新这篇文章。谢谢@CommonsWare
  • 我的意思是这种行为是无证的,特别是考虑到我们在活动中没有看到任何其他内容。 findViewById() 可能会为 R.id.img 返回 null,也可能不会。可以想象,这种行为可能会因设备而异,具体取决于 Android 操作系统版本、设备制造商更改和 ROM 模块。
  • “我觉得很奇怪,文档没有说明在这种情况下应该做什么”——我会先查看findViewById() 的返回值,而不是盲目地查看假设成功。我还强烈考虑使用保留的片段来管理您的 AsyncTasks,或者以其他方式教您的 AsyncTask 在后台线程运行时活动可以来去去去(例如,用户旋转屏幕)。
  • 听起来很合理。

标签: android android-asynctask background-process ui-thread


【解决方案1】:

考虑这个任务(其中 R.id.test 指的是我的活动布局中的有效视图):

public class LongTaskTest extends AsyncTask<Void, Void, Void>{
    private WeakReference<Activity> mActivity;
    public LongTaskTest(Activity a){
        mActivity = new WeakReference<Activity>(a);
    }
    @Override protected Void doInBackground(Void... params) {
        LogUtil.d("LongTaskTest.doInBackground()");
        SystemClock.sleep(5*60*1000);
        LogUtil.d("mActivity.get()==null " + (mActivity.get()==null));
        LogUtil.d("mActivity.get().findViewById(R.id.frame)==null " + (mActivity.get().findViewById(R.id.test)==null));
        return null;
    }
}

如果我像这样从 Activity 的 onCreate 运行此任务:

public class Main extends Activity {
    @Override
    public void onCreate(Bundle state) {
        super.onCreate(state);
        setContentView(R.layout.testlayout);
        new LongTaskTest(this).execute();
        finish();
    }
}

无论我在后台线程休眠多久,我的日志总是显示:

LongTaskTest.doInBackground()
mActivity.get()==null false
mActivity.get().findViewById(R.id.frame)==null false

也就是说,活动及其视图似乎保持活跃(即使我通过 DDMS 手动发出 GC)。如果我有更多时间,我会查看内存转储,但否则我真的不知道为什么会出现这种情况......但在回答你的问题时,似乎:

  • 它会静默失败吗?没有
  • 它会产生任何异常吗?没有
  • 是否会通知用户出现问题?没有

【讨论】:

  • 活动可能仍然存在,因为你的内部类有对它的引用,但是其中的视图和实例变量呢?
  • 这就是为什么我对 Activity 使用 Wea​​kReference ... 以确保我的内部类不是 Activity 保持活跃的原因。至于视图和实例变量,我的内部类除了对 Activity 的 WeakReference 之外什么都没有。除非你的意思是别的?更多关于弱参考:developer.android.com/reference/java/lang/ref/…
  • @newbyca 如果事实上 LongTaskTest 是一个内部类,那么它有一个你忘记的对 Activity 的隐式引用 - 让它成为一个静态嵌套类:stackoverflow.com/questions/70324/…
【解决方案2】:

即使您的 Activity 被破坏(即您的主线程被破坏),doInBackground() 也会继续运行,因为 doInBackground() 方法在工作线程/后台线程上运行。运行 onPostExecute() 方法时会出现“问题”,因为它在主/UI 线程上运行,您可能会遇到不相关的数据,但不会向用户显示异常。因此,最好在 Activity 被销毁时取消 AsyncTask,因为当 Activity 不再存在时没有理由运行 AsyncTask。如果即使组件/活动被破坏,您仍想继续从网络下载内容,请使用 android 服务。谢谢。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-03-06
    • 1970-01-01
    • 2012-03-23
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多