【Android】永不消失的Toast

自己亲测两种解决方法:

1、将Toast换成dialog,呵呵,这个方法比较土鳖,但是很好用。而且不用费力去理原有代码的流程。

原因:Toast在传入context的时候,不管你传的是activity还是application,他都是调用的mContext.getPackageName();这也是为什么只有卸载应用才能将Toast消除。但是dialog就不一样,他是依赖activity本身的,如果activity退出了,activity附属的dialog也就自然被回收了。补充,当由于HandlerThread没quiet,而activity退出的时候,Toast无法调用相应的hide方法,导致Toast不消失。

public void show() {
        if (mNextView == null) {
            throw new RuntimeException("setView must have been called");
        }

        INotificationManager service = getService();
        String pkg = mContext.getPackageName();
        TN tn = mTN;
        tn.mNextView = mNextView;

        try {
            service.enqueueToast(pkg, tn, mDuration);
        } catch (RemoteException e) {
            // Empty
        }
    }



2、方法二就是常规的处理方式,在onpause方法里也加个HandlerThread.quiet.

Toast是Android一个比较省心的控件,因为Toast不提供任何交互界面,看一眼就消失,而且只需要Context就可以创建。

然而无须交互的优势这也带来一个问题,如果Toast显示后无法消失,那么将真的是无法消失了。除非强制退出应用或者重启手机。

 

一个简单的示例程序,就可以创造出一个无法消失的Toast。以下为Activity代码:

public class ToastTestActivity extends Activity {
	/** HandlerThread object */
    HandlerThread mThread = null;
    
    /** Handler object */
    ToastHandler mHandler = null;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        mThread = new HandlerThread("ToastThread");
        mThread.start();
        mHandler = new ToastHandler(mThread.getLooper());
    }

    /**
     * Called when Button got clicked
     * @param v View that got clicked
     */
    public void myClick(View v) {
        mHandler.sendEmptyMessageDelayed(1, 2000);
    }

    private class ToastHandler extends Handler {
        public ToastHandler(Looper looper) {
            super(looper);
        }

        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            Toast toast = Toast.makeText(ToastTestActivity.this, "Dismiss me", Toast.LENGTH_LONG);
            toast.show();
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        mHandler.removeMessages(1);
        mThread.getLooper().quit();
        mHandler = null;
        mThread = null;
    }
}
 

该Activity有一个Button, 点击之后会向HandlerThread发送一个延时消息,Handler处理该消息时会创建并显示一个Toast,当Toast正在显示的时候,点Back键退出Activity。就会发现正在显示的Toast不会消失。

 

原因:

罪魁祸首就是HandlerThread在onDestroy方法里面调用了HandlerThread.getLooper().quit()。因为Toast在创建的时候,会同时创建一个基于当前线程的Handler对象,虽然NotificationManagerService负责调度Toast,但是真正显示和隐藏Toast都是向这个Handler发送消息。也就是说Toast做事情也用到了HandlerThread的Looper。所以如果单纯在onDestroy方法里将Looper给quit的话,Toast将无法实现隐藏。

解决方案:

个人感觉让Toast远离HandlerThread比较靠谱。可以统一规范代码,所有显示Toast的代码都post到主线程(无论是Activity还是Service)当中,因为主线程的Looper不会轻易quit,至少不会允许开发者显示quit。

你可能感兴趣的:(BUG解决)