自己亲测两种解决方法:
1、将Toast换成dialog,呵呵,这个方法比较土鳖,但是很好用。而且不用费力去理原有代码的流程。
原因:Toast在传入context的时候,不管你传的是activity还是application,他都是调用的mContext.getPackageName();这也是为什么只有卸载应用才能将Toast消除。但是dialog就不一样,他是依赖activity本身的,如果activity退出了,activity附属的dialog也就自然被回收了。补充<span style="font-family: Arial; font-size: 16px; line-height: 24px; background-color: rgb(237, 237, 237);">,当由于</span><span style="font-family: Arial; font-size: 16px; line-height: 24px; background-color: rgb(237, 237, 237);">HandlerThread没quiet,而activity退出的时候,Toast无法调用相应的hide方法,导致Toast不消失。</span>
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。