dialog的setCancelable与setCanceledOnTouchOutside的区别、dismiss与cancel和hide区别、异常等

本文深入解析Android Dialog中的cancel、dismiss和hide方法的区别,同时对比setCancelable与setCanceledOnTouchOutside的用法,提供避免常见错误的解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

看源码:


/**
 * Sets whether this dialog is cancelable with the
 * {@link KeyEvent#KEYCODE_BACK BACK} key. //点击物理返回键是否可取消dialog
 */
public void setCancelable(boolean flag) {
    mCancelable = flag;
}

/**
 * Sets whether this dialog is canceled when touched outside the window's
 * bounds. If setting to true, the dialog is set to be cancelable if not
 * already set.
 * 
 * @param cancel Whether the dialog should be canceled when touched outside
 *            the window.   //点击dialog之外 是否可取消
 */
public void setCanceledOnTouchOutside(boolean cancel) {
    if (cancel && !mCancelable) {
        mCancelable = true;
    }

    mWindow.setCloseOnTouchOutside(cancel);
}

/**
 * Cancel the dialog.  This is essentially the same as calling {@link #dismiss()}, but it will
 * also call your {@link DialogInterface.OnCancelListener} (if registered).
 *  // 取消dialog ,和dismiss相同,但cancel会回调DialogInterface.OnCancelListener
 */
@Override
public void cancel() {
    if (!mCanceled && mCancelMessage != null) {
        mCanceled = true;
        // Obtain a new message so this dialog can be re-used
        Message.obtain(mCancelMessage).sendToTarget();
    }
    dismiss();
}

/**
 * Set a listener to be invoked when the dialog is canceled.    //设置监听取消dialog
 *
 * <p>This will only be invoked when the dialog is canceled.
 * Cancel events alone will not capture all ways that
 * the dialog might be dismissed. If the creator needs
 * to know when a dialog is dismissed in general, use
 * {@link #setOnDismissListener}.</p>
 * 
 * @param listener The {@link DialogInterface.OnCancelListener} to use.
 */
public void setOnCancelListener(@Nullable OnCancelListener listener) {
    if (mCancelAndDismissTaken != null) {
        throw new IllegalStateException(
                "OnCancelListener is already taken by "
                + mCancelAndDismissTaken + " and can not be replaced.");
    }
    if (listener != null) {
        mCancelMessage = mListenersHandler.obtainMessage(CANCEL, listener);
    } else {
        mCancelMessage = null;
    }
}

/**
 * Hide the dialog, but do not dismiss it.  //隐藏dialog(设置为Gone),但是不dismiss
 *      //hide只是把对话框里的DecorView设置为不可见,并没有从Window移除掉这个DecorView。
 */
public void hide() {
    if (mDecor != null) {
        mDecor.setVisibility(View.GONE);
    }
}

/**
 * Dismiss this dialog, removing it from the screen. This method can be
 * invoked safely from any thread.  Note that you should not override this
 * method to do cleanup when the dialog is dismissed, instead implement
 * that in {@link #onStop}. //从屏幕移除,可以任意线程使用,在onstop中移除
 *      //dismiss可以在任何线程调用,但是最好不要覆写dismiss方法,实在需要就在onStop里去dismiss。
 */
@Override
public void dismiss() {
    if (Looper.myLooper() == mHandler.getLooper()) {
        dismissDialog();
    } else {
        mHandler.post(mDismissAction);
    }
}

dismiss、cancel和hide区别

dismiss、cancel两者效果都是一样的,其实看下源码就知道cancel肯定会去调dismiss的,如果调用的cancel的话就可以监听DialogInterface.OnCancelListener。

hide只是隐藏了对话框并没有销毁,如果打算用这方法来灭掉对话框就会出现问题,在Activity销毁的时候就会出现崩溃日志了,因为Activity销毁时是需要把对话框都关闭掉的。

setCancelable与setCanceledOnTouchOutside的区别

setCancelable : 点击物理返回键是否可关闭弹框

setCanceledOnTouchOutside: 点击弹框外 是否可关闭弹框

异常:

not attached to window manager

解决:

ondestroy中要关闭dialog:

  @Override
    protected void onDestroy() {
        if (progressDialog != null && progressDialog.isShowing()) {
            progressDialog.dismiss();
            progressDialog = null;
        }
        super.onDestroy();
    }

示例如下:

public class TestActivity extends AppCompatActivity {

    private Dialog progressDialog;

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

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

    @Override
    protected void onDestroy() {
        if (progressDialog != null && progressDialog.isShowing()) {
            progressDialog.dismiss();
        }
        progressDialog = null;

        super.onDestroy();
    }

    protected void showLoading() {
        if (progressDialog == null) {
            progressDialog = new Dialog(this, R.style.progress_dialog);
        }

        if (progressDialog.isShowing()) {
            progressDialog.dismiss();
        }
        progressDialog.setContentView(R.layout.logindialog);
        progressDialog.setCancelable(true);
        progressDialog.setCanceledOnTouchOutside(false);
        Window window = progressDialog.getWindow();
        if (window != null) {
            window.setBackgroundDrawableResource(android.R.color.transparent);
        }
        progressDialog.show();
    }

    protected void dismissLoading() {
        if (progressDialog != null && progressDialog.isShowing()) {
            progressDialog.dismiss();
        }
    }

}
<think>好的,我现在需要回答用户关于Android中dialog.dismiss()dialog.cancel()的区别及用法的问题。首先,我得回忆一下这两个方法的区别。记得dismiss()是直接关闭对话框,而cancel()在关闭前会触发取消事件,比如调用OnCancelListener。但具体的细节可能需要进一步确认。 我应该先查阅Android的官方文档,确认这两个方法的具体行为。根据文档,dismiss()方法仅仅是移除对话框,不会触发任何取消事件。而cancel()方法在调用时,除了移除对话框,还会触发OnCancelListener,如果有设置的话。此外,如果对话框是通过show()方法显示的,cancel()方法可能还会发送一个取消的结果,这在某些情况下可能影响后续的逻辑。接下来,我需要考虑如何用示例代码来说明两者的用法。例如,创建一个对话框并设置OnCancelListener,当调用cancel()时,这个监听器会被触发,而dismiss()则不会。同时,可能需要说明在哪些场景下使用哪个方法更合适,比如当需要处理取消操作时使用cancel(),否则使用dismiss()。另外,用户可能还想知道这两个方法在实际开发中的常见错误或注意事项。比如,重复调用dismiss()是否安全,或者是否需要在取消后做一些资源清理工作。这时候需要指出,通常多次调用dismiss()不会有问题,但最好通过判断对话框是否还在显示来避免异常。 还要考虑是否有相关的引用资料可以支持回答。例如,引用中提到的BootstrapDialog的实例操作,虽然来自不同的框架,但概念上可能类似,可以辅助说明。但主要还是要依据Android本身的文档常见实践。 最后,组织回答的结构,先解释区别,再给出示例代码,然后说明使用场景注意事项,确保内容清晰易懂。同时,生成相关问题,帮助用户进一步探索,比如性能影响或生命周期问题。</think>### dialog.dismiss() dialog.cancel() 的区别及用法 在 Android 中,`dialog.dismiss()` `dialog.cancel()` 都用于关闭对话框,但行为细节适用场景不同: --- #### **1. 核心区别** | 方法 | 行为 | 触发监听器 | |-----------------|----------------------------------------------------------------------|--------------------------| | `dismiss()` | 直接关闭对话框,**不触发**取消相关的逻辑 | 无 | | `cancel()` | 关闭对话框,**触发**取消相关的逻辑(如 `OnCancelListener`) | 触发 `OnCancelListener` | --- #### **2. 代码示例** ```java // 创建对话框并设置监听器 AlertDialog dialog = new AlertDialog.Builder(context) .setTitle("示例对话框") .setPositiveButton("确定", null) .setOnCancelListener(dialogInterface -> { Log.d("TAG", "对话框被取消"); // 仅通过 cancel() 触发 }) .create(); // 场景1:调用 dismiss() dialog.dismiss(); // 直接关闭,无日志输出 // 场景2:调用 cancel() dialog.cancel(); // 关闭并输出日志:"对话框被取消" ``` --- #### **3. 使用场景** - **`dismiss()`** - 适用于**明确关闭对话框**(如用户点击“确定”按钮)。 - 需要避免触发取消逻辑时使用。 ```java positiveButton.setOnClickListener(v -> dialog.dismiss()); ``` - **`cancel()`** - 适用于**用户主动取消操作**(如点击返回键或外部区域)。 - 需要清理取消相关资源时使用。 ```java dialog.setOnKeyListener((d, keyCode, event) -> { if (keyCode == KeyEvent.KEYCODE_BACK) { dialog.cancel(); // 触发 OnCancelListener return true; } return false; }); ``` --- #### **4. 注意事项** - **重复调用安全**:多次调用 `dismiss()` 是安全的,但建议先检查 `dialog.isShowing()`。 - **生命周期关联**:若对话框依赖 Activity 上下文,需在 `onDestroy()` 中调用 `dismiss()` 避免内存泄漏。 - **异步操作**:在异步任务完成后调用 `dismiss()`,需通过 `runOnUiThread()` 确保线程安全。 --- #### **5. 底层机制** - `cancel()` 内部会调用 `dismiss()`,但增加了事件触发逻辑[^3]。 - 若对话框通过 `show()` 显示,`cancel()` 可能发送 `RESULT_CANCELED` 到关联的 `DialogInterface.OnCancelListener`[^3]。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值