9

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

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")
    }
}

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

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

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

4

2 回答 2

3

考虑这个任务(其中 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)。如果我有更多时间,我会查看内存转储,但否则我真的不知道为什么会这样......但在回答你的问题时,似乎:

  • 它会默默地失败吗?不
  • 它会产生任何异常吗?不
  • 是否会通知用户出现问题?不
于 2012-09-04T16:10:18.690 回答
2

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

于 2015-04-18T10:21:33.283 回答