창 관리자 충돌에 첨부되지 않은보기 protected void onPreExecute()

ACRA를 사용하여 앱 충돌을보고합니다. 내가지고 있다고 View not attached to window manager오류 메시지와 나는 포장하여 고정했다고 생각 pDialog.dismiss();if 문에서 :

if (pDialog!=null)
{
    if (pDialog.isShowing())
    {
        pDialog.dismiss();
    }
}

View not attached to window manager수신 되는 충돌 의 양이 줄어들 었지만 여전히 일부 문제가 발생하여 해결 방법을 잘 모르겠습니다.

에러 메시지:

java.lang.IllegalArgumentException: View not attached to window manager
at android.view.WindowManagerGlobal.findViewLocked(WindowManagerGlobal.java:425)
at android.view.WindowManagerGlobal.removeView(WindowManagerGlobal.java:327)
at android.view.WindowManagerImpl.removeView(WindowManagerImpl.java:83)
at android.app.Dialog.dismissDialog(Dialog.java:330)
at android.app.Dialog.dismiss(Dialog.java:312)
at com.package.class$LoadAllProducts.onPostExecute(class.java:624)
at com.package.class$LoadAllProducts.onPostExecute(class.java:1)
at android.os.AsyncTask.finish(AsyncTask.java:631)
at android.os.AsyncTask.access$600(AsyncTask.java:177)
at android.os.AsyncTask$InternalHandler.handleMessage(AsyncTask.java:644)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:176)
at android.app.ActivityThread.main(ActivityThread.java:5419)
at java.lang.reflect.Method.invokeNative(Native Method)
at java.lang.reflect.Method.invoke(Method.java:525)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1046)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:862)
at dalvik.system.NativeStart.main(Native Method)

코드 스 니펫 :

class LoadAllProducts extends AsyncTask<String, String, String>
{

    /**
     * Before starting background thread Show Progress Dialog
     * */
    @Override
    protected void onPreExecute()
    {
        super.onPreExecute();
        pDialog = new ProgressDialog(CLASS.this);
        pDialog.setMessage("Loading. Please wait...");
        pDialog.setIndeterminate(false);
        pDialog.setCancelable(false);
        pDialog.show();
    }

    /**
     * getting All products from url
     * */
    protected String doInBackground(String... args)
    {
        // Building Parameters
        doMoreStuff("internet");
        return null;
    }


    /**
     * After completing background task Dismiss the progress dialog
     * **/
    protected void onPostExecute(String file_url)
    {
         // dismiss the dialog after getting all products
         if (pDialog!=null)
         {
                if (pDialog.isShowing())
                {
                    pDialog.dismiss();   //This is line 624!    
                }
         }
         something(note);
    }
}

명백한:

    <activity
        android:name="pagename.CLASS"
        android:configChanges="keyboard|keyboardHidden|orientation|screenSize|screenLayout"
        android:label="@string/name" >
    </activity>

이 충돌이 발생하지 않도록 무엇을 놓치고 있습니까?



답변

버그를 재현하는 방법 :

  1. 장치에서이 옵션을 활성화하십시오 : Settings -> Developer Options -> Don't keep Activities.
  2. AsyncTask가 실행 중이고 ProgressDialog가 표시된 상태에서 홈 버튼을 누릅니다 .

Android OS는 숨겨 지 자마자 활동을 파괴합니다. 때 onPostExecute호출되는 Activity될 것 “마무리” 상태와 ProgressDialog부착되지는 않습니다 Activity.

해결 방법 :

  1. onPostExecute분석법 에서 활동 상태를 확인하십시오 .
  2. ProgressDialogin onDestroy메소드를 해제하십시오 . 그렇지 않으면 android.view.WindowLeaked예외가 발생합니다. 이 예외는 일반적으로 활동이 완료 될 때 여전히 활성 인 대화 상자에서 발생합니다.

이 고정 코드를 사용해보십시오 :

public class YourActivity extends Activity {

    private void showProgressDialog() {
        if (pDialog == null) {
            pDialog = new ProgressDialog(StartActivity.this);
            pDialog.setMessage("Loading. Please wait...");
            pDialog.setIndeterminate(false);
            pDialog.setCancelable(false);
        }
        pDialog.show();
    }

    private void dismissProgressDialog() {
        if (pDialog != null && pDialog.isShowing()) {
            pDialog.dismiss();
        }
    }

    @Override
    protected void onDestroy() {
        dismissProgressDialog();
        super.onDestroy();
    }

    class LoadAllProducts extends AsyncTask<String, String, String> {

        // Before starting background thread Show Progress Dialog
        @Override
        protected void onPreExecute() {
            showProgressDialog();
        }

        //getting All products from url
        protected String doInBackground(String... args) {
            doMoreStuff("internet");
            return null;
        }

        // After completing background task Dismiss the progress dialog
        protected void onPostExecute(String file_url) {
            if (YourActivity.this.isDestroyed()) { // or call isFinishing() if min sdk version < 17
                return;
            }
            dismissProgressDialog();
            something(note);
        }
    }
}


답변

문제는 그 수 Activity있었다 finishedprogress of finishing.

확인을 추가 isFinishing하고이 경우에만 대화 상자를 닫습니다.false

if (!YourActivity.this.isFinishing() && pDialog != null) {
    pDialog.dismiss();
}

isFinishing : 이 활동이 완료 중인지 확인하십시오. 호출 finish했거나 다른 사람이 완료를 요청했기 때문입니다.


답변

들어 DialogA의 생성 Fragment, 나는 다음과 같은 코드를 사용 :

ProgressDialog myDialog = new ProgressDialog(getActivity());
myDialog.setOwnerActivity(getActivity());
...
Activity activity = myDialog.getOwnerActivity();
if( activity!=null && !activity.isFinishing()) {
    myDialog.dismiss();
}

이 패턴을 사용하여 Fragment가에서 분리 될 수 있는 경우를 처리 합니다 Activity.


답변

코드가 어떻게 작동하는지보십시오 :

비동기 작업을 호출 한 후 백그라운드에서 비동기 작업이 실행됩니다. 그것은 바람직하다. 이제이 비동기 작업에는 코드를 보는 방법을 묻는 경우 활동에 첨부 된 진행률 대화 상자가 있습니다.

pDialog = new ProgressDialog(CLASS.this);

당신은 전달되는 Class.this인수에 컨텍스트한다. 따라서 진행 대화 상자가 여전히 활동에 첨부되어 있습니다.

이제 시나리오를 고려하십시오. 비동기 작업이 진행되는 동안 finish () 메소드를 사용하여 활동을 완료하려고하면 활동에 첨부 된 자원에 액세스하려고하는 시점이 progress bar됩니다 ( 예 : 활동이 더 이상 없을 때) 그곳에.

따라서 당신은 얻는다 :

java.lang.IllegalArgumentException: View not attached to the window manager

이것에 대한 해결책 :

1) 활동이 완료되기 전에 대화 상자가 닫히거나 취소되었는지 확인하십시오.

2) 대화 상자가 닫히고 비동기 작업이 끝난 후에 만 ​​활동을 완료하십시오.


답변

@erakitin 답변을 기반으로하지만 Android 버전 <API 레벨 17 과도 호환됩니다. Sadly Activity.isDestroyed () 는 API 레벨 17 이후에만 지원되므로 나처럼 이전 API 레벨을 대상으로하는 경우 직접 확인하십시오. View not attached to window manager그 후에 예외 가 없습니다 .

예제 코드

public class MainActivity extends Activity {
    private TestAsyncTask mAsyncTask;
    private ProgressDialog mProgressDialog;
    private boolean mIsDestroyed;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (condition) {
            mAsyncTask = new TestAsyncTask();
            mAsyncTask.execute();
        }
    }

    @Override
    protected void onResume() {
        super.onResume();

        if (mAsyncTask != null && mAsyncTask.getStatus() != AsyncTask.Status.FINISHED) {
            Toast.makeText(this, "Still loading", Toast.LENGTH_LONG).show();
            return;
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mIsDestroyed = true;

        if (mProgressDialog != null && mProgressDialog.isShowing()) {
            mProgressDialog.dismiss();
        }
    }

    public class TestAsyncTask extends AsyncTask<Void, Void, AsyncResult> {
        @Override
        protected void onPreExecute() {
            super.onPreExecute();
            mProgressDialog = ProgressDialog.show(MainActivity.this, "Please wait", "doing stuff..");
        }

        @Override
        protected AsyncResult doInBackground(Void... arg0) {
            // Do long running background stuff
            return null;
        }

        @Override
        protected void onPostExecute(AsyncResult result) {
            // Use MainActivity.this.isDestroyed() when targeting API level 17 or higher
            if (mIsDestroyed)// Activity not there anymore
                return;

            mProgressDialog.dismiss();
            // Handle rest onPostExecute
        }
    }
}


답변

@Override
public void onPause() {
    super.onPause();

    if(pDialog != null)
        pDialog .dismiss();
    pDialog = null;
}

이것을 참조 하십시오 .


답변

ConfigurationChanged에서 재정의하고 진행률 대화 상자를 닫습니다. 진행률 대화 상자가 세로로 작성되고 가로로 닫히면 창 관리자에 첨부되지 않은보기 오류가 발생합니다.

또한 진행률 표시 줄을 중지하고 onPause (), onBackPressed 및 onDestroy 메소드에서 비동기 작업을 중지하십시오.

if(asyncTaskObj !=null && asyncTaskObj.getStatus().equals(AsyncTask.Status.RUNNING)){

    asyncTaskObj.cancel(true);

}