Background task, progress dialog, orientation change - is there any 100% working solution?

后端 未结 8 2138
半阙折子戏
半阙折子戏 2020-11-22 06:37

I download some data from internet in background thread (I use AsyncTask) and display a progress dialog while downloading. Orientation changes, Activity is rest

相关标签:
8条回答
  • 2020-11-22 07:16

    While Mark's (CommonsWare) answer does indeed work for orientation changes, it fails if the Activity is destroyed directly (like in the case of a phone call).

    You can handle the orientation changes AND the rare destroyed Activity events by using an Application object to reference your ASyncTask.

    There's an excellent explanation of the problem and the solution here:

    Credit goes completely to Ryan for figuring this one out.

    0 讨论(0)
  • 2020-11-22 07:16

    After 4 years Google solved the problem just calling setRetainInstance(true) in Activity onCreate. It will preserve your activity instance during device rotation. I have also a simple solution for older Android.

    0 讨论(0)
  • 2020-11-22 07:18

    Step #1: Make your AsyncTask a static nested class, or an entirely separate class, just not an inner (non-static nested) class.

    Step #2: Have the AsyncTask hold onto the Activity via a data member, set via the constructor and a setter.

    Step #3: When creating the AsyncTask, supply the current Activity to the constructor.

    Step #4: In onRetainNonConfigurationInstance(), return the AsyncTask, after detaching it from the original, now-going-away activity.

    Step #5: In onCreate(), if getLastNonConfigurationInstance() is not null, cast it to your AsyncTask class and call your setter to associate your new activity with the task.

    Step #6: Do not refer to the activity data member from doInBackground().

    If you follow the above recipe, it will all work. onProgressUpdate() and onPostExecute() are suspended between the start of onRetainNonConfigurationInstance() and the end of the subsequent onCreate().

    Here is a sample project demonstrating the technique.

    Another approach is to ditch the AsyncTask and move your work into an IntentService. This is particularly useful if the work to be done may be long and should go on regardless of what the user does in terms of activities (e.g., downloading a large file). You can use an ordered broadcast Intent to either have the activity respond to the work being done (if it is still in the foreground) or raise a Notification to let the user know if the work has been done. Here is a blog post with more on this pattern.

    0 讨论(0)
  • 2020-11-22 07:18

    The accepted answer was very helpful, but it doesn't have a progress dialog.

    Fortunately for you, reader, I have created an extremely comprehensive and working example of an AsyncTask with a progress dialog!

    1. Rotation works, and the dialog survives.
    2. You can cancel the task and dialog by pressing the back button (if you want this behaviour).
    3. It uses fragments.
    4. The layout of the fragment underneath the activity changes properly when the device rotates.
    0 讨论(0)
  • 2020-11-22 07:19

    I've toiled for a week to find a solution to this dilemma without resorting to editing the manifest file. The assumptions for this solution are:

    1. You always need to use a progress dialog
    2. Only one task is performed at a time
    3. You need the task to persist when the phone is rotated and the progress dialog to be automatically dismisses.

    Implementation

    You will need to copy the two files found at the bottom of this post into your workspace. Just make sure that:

    1. All your Activitys should extend BaseActivity

    2. In onCreate(), super.onCreate() should be called after you initialize any members that need to be accessed by your ASyncTasks. Also, override getContentViewId() to provide the form layout id.

    3. Override onCreateDialog() like usual to create dialogs managed by the activity.

    4. See code below for a sample static inner class to make your AsyncTasks. You can store your result in mResult to access later.


    final static class MyTask extends SuperAsyncTask<Void, Void, Void> {
    
        public OpenDatabaseTask(BaseActivity activity) {
            super(activity, MY_DIALOG_ID); // change your dialog ID here...
                                           // and your dialog will be managed automatically!
        }
    
        @Override
        protected Void doInBackground(Void... params) {
    
            // your task code
    
            return null;
        }
    
        @Override
        public boolean onAfterExecute() {
            // your after execute code
        }
    }
    

    And finally, to launch your new task:

    mCurrentTask = new MyTask(this);
    ((MyTask) mCurrentTask).execute();
    

    That's it! I hope this robust solution will help someone.

    BaseActivity.java (organize imports yourself)

    protected abstract int getContentViewId();
    
    public abstract class BaseActivity extends Activity {
        protected SuperAsyncTask<?, ?, ?> mCurrentTask;
        public HashMap<Integer, Boolean> mDialogMap = new HashMap<Integer, Boolean>();
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
    
            setContentView(getContentViewId());
    
            mCurrentTask = (SuperAsyncTask<?, ?, ?>) getLastNonConfigurationInstance();
            if (mCurrentTask != null) {
                mCurrentTask.attach(this);
                if (mDialogMap.get((Integer) mCurrentTask.dialogId) != null
                    && mDialogMap.get((Integer) mCurrentTask.dialogId)) {
            mCurrentTask.postExecution();
                }
            }
        }
    
        @Override
        protected void onPrepareDialog(int id, Dialog dialog) {
        super.onPrepareDialog(id, dialog);
    
            mDialogMap.put(id, true);
        }
    
        @Override
        public Object onRetainNonConfigurationInstance() {
            if (mCurrentTask != null) {
                mCurrentTask.detach();
    
                if (mDialogMap.get((Integer) mCurrentTask.dialogId) != null
                    && mDialogMap.get((Integer) mCurrentTask.dialogId)) {
                    return mCurrentTask;
                }
            }
    
            return super.onRetainNonConfigurationInstance();
        }
    
        public void cleanupTask() {
            if (mCurrentTask != null) {
                mCurrentTask = null;
                System.gc();
            }
        }
    }
    

    SuperAsyncTask.java

    public abstract class SuperAsyncTask<Params, Progress, Result> extends AsyncTask<Params, Progress, Result> {
        protected BaseActivity mActivity = null;
        protected Result mResult;
        public int dialogId = -1;
    
        protected abstract void onAfterExecute();
    
        public SuperAsyncTask(BaseActivity activity, int dialogId) {
            super();
            this.dialogId = dialogId;
            attach(activity);
        }
    
        @Override
        protected void onPreExecute() {
            super.onPreExecute();
            mActivity.showDialog(dialogId); // go polymorphism!
        }    
    
        protected void onPostExecute(Result result) {
            super.onPostExecute(result);
            mResult = result;
    
            if (mActivity != null &&
                    mActivity.mDialogMap.get((Integer) dialogId) != null
                    && mActivity.mDialogMap.get((Integer) dialogId)) {
                postExecution();
            }
        };
    
        public void attach(BaseActivity activity) {
            this.mActivity = activity;
        }
    
        public void detach() {
            this.mActivity = null;
        }
    
        public synchronized boolean postExecution() {
            Boolean dialogExists = mActivity.mDialogMap.get((Integer) dialogId);
            if (dialogExists != null || dialogExists) {
                onAfterExecute();
                cleanUp();
        }
    
        public boolean cleanUp() {
            mActivity.removeDialog(dialogId);
            mActivity.mDialogMap.remove((Integer) dialogId);
            mActivity.cleanupTask();
            detach();
            return true;
        }
    }
    
    0 讨论(0)
  • 2020-11-22 07:24

    This is my solution: https://github.com/Gotchamoh/Android-AsyncTask-ProgressDialog

    Basically the steps are:

    1. I use onSaveInstanceState to save the task if it is still processing.
    2. In onCreate I get the task if it was saved.
    3. In onPause I discard the ProgressDialog if it is shown.
    4. In onResume I show the ProgressDialog if the task is still processing.
    0 讨论(0)
提交回复
热议问题