前言
在 Android 11 Toast 的行为发生了变更
![](https://i-blog.csdnimg.cn/blog_migrate/ed2d1799f00c70fcf3cdbfe75194f35a.png)
禁止后台自定义 Toast
text toast 不允许自定义
setView() 被弃用
新增 Toast.Callback 回调
Android 11 API 变更
禁止后台自定义 Toast
自定义 Toast 「不能」 在 app 处于后台时显示,取而代之会显示 「"Background custom toast blocked for package [packageName] See g.co/dev/toast."」 的文本 toast
![](https://i-blog.csdnimg.cn/blog_migrate/de75b59f5da95d1d938b730735852719.png)
禁止后台自定义 Toast
普通的 text toast
不受影响
![](https://i-blog.csdnimg.cn/blog_migrate/e806529f86c2e7281798404e2c916dc2.png)
普通的 text toast 不受影响
text toast 不允许自定义
默认的 toast 是 text toast
,如果想使用自定义的 toast ,需要调用 setView() 方法
在 targetSdkVersion 为 R 或更高时,调用 setGravity 和 setMargin 方法将不进行任何操作
❝
官方文档中所述的 Android R 仅影响 「text toast」 ,而自定义的 toast 不受影响
❞
![](https://i-blog.csdnimg.cn/blog_migrate/dba7fd3ed8529a25ffaf1c27d0783b3c.png)
调用无效,仅影响 text toast
![](https://i-blog.csdnimg.cn/blog_migrate/a0f24fbb31bac37111511a476054a670.png)
调用无效,仅影响 test toast
如图,在 test toast 中调用 setGravity 和 setMargin 方法,但 toast 位置并未居中
![](https://i-blog.csdnimg.cn/blog_migrate/87f8910af88b38579c9803d3bee0de3f.png)
在 test toast 中调用 setGravity 和 setMargin 方法
![](https://i-blog.csdnimg.cn/blog_migrate/0ecc06ed42365891ad3c67fa95105c73.png)
并未居中,方法不生效
setView() 被弃用
setView() 方法被标记弃用
❝
「Deprecated」 表示该功能目前仍可以使用,但可能会在将来的 Android 版本中删除。建议开发人员避免长期使用此功能
❞
![](https://i-blog.csdnimg.cn/blog_migrate/7dad3a5e4b8f35f39efdc22c23c336fc.png)
setView 被弃用
可以看到,官方在一步步禁止自定义 Toast
目前是 targetSdkVersion 为 R 或更高的 app 禁止后台弹出自定义 Toast
同时将 setView() 方法标记弃用,当该方法从源码中移除后,自定 Toast 的方式将被彻底消灭
❝
当然,官方提供了相应的替代品,使用 Snackbar
❞
新增 Toast.Callback 回调
添加了新的回调(Toast.Callback
),以通知 Toast 显示和隐藏。可以通过以下方法轻松将其添加到 Toast 中:
val toast = Toast.makeText(this, R.string.simple2_toast, Toast.LENGTH_SHORT)toast.addCallback(object : Toast.Callback() { override fun onToastShown() { super.onToastShown() Log.d(TAG, "onToastShown") } override fun onToastHidden() { super.onToastHidden() Log.d(TAG, "onToastHidden") }})toast.show()
一些小 tips 及 demo
demo 在这 ,切换 Flavor 即可指定不同的 targetSdkVersion
![](https://i-blog.csdnimg.cn/blog_migrate/8595c2a1985f91315888e255afaa8b3a.png)
切换 Flavor
在写 demo 时遇到一些小问题
tip1
Handler()
无参构造方法和 Handler(Handler.Callback)
构造方法 被弃用了
![](https://i-blog.csdnimg.cn/blog_migrate/b38f6b801d6da088fb6a97b2b006eee7.png)
无参构造器被弃用
简单来讲就是在初始化 Handler 时要显示的配置 Looper
Handler 使用不当会有这样一种 bug,例如在子线程通过无参构造函数创建 Handler,您可能会看到这样的异常
![](https://i-blog.csdnimg.cn/blog_migrate/b661be7764fa079cc2c04f32f4d7e117.png)
错误日志
![](https://i-blog.csdnimg.cn/blog_migrate/54e230bb2e8fdb79763bac34749418fe.png)
抛出异常源码
详细内容这里就不讲了,这是 Android 开发者的必备知识
官方通过强制使用传入 Looper 的 Handler 构造器来避免使用中的问题
tip2
过去使用 Toast 构造器创建 Toast 对象 并调用 setText 方法会崩溃,targetSdkVersion 为 R 时不会崩溃
![](https://i-blog.csdnimg.cn/blog_migrate/6538ef2186e333758db2fb958e5c6d1a.png)
相同的代码 targetSdkVersion 低版本会崩溃
![](https://i-blog.csdnimg.cn/blog_migrate/df0fa586d33299d8e897b30a5438847f.png)
崩溃,但设置位置生效
![](https://i-blog.csdnimg.cn/blog_migrate/2e0211c4d98a59fbf423167ab9e77c62.png)
异常log
![](https://i-blog.csdnimg.cn/blog_migrate/0724453b5c5c471f325dd776324f2723.png)
API 29 源码
API 29 中调用 setText() 方法时要保证 mNextView 不为空,而 mNextView 是调用 setView 赋值的
![](https://i-blog.csdnimg.cn/blog_migrate/119f10b2f0b99bdfffc59b2c33788ac7.png)
API 29 setView 源码
因此过去使用 Toast 构造器创建 toast 对象无法创建普通的 text toast,必须调用 setView 方法
至于 API 30 肯定在这里做了修改,由于现在看不到源码,我也猜测不出官方的用意
如果各位小伙伴有什么想法欢迎评论区留言
相关阅读
Android 7.x Toast BadTokenException处理
欢迎关注微信公众号 code小生,专注移动开发、大前端技术、Google等资讯,扫描下方二维码立刻关注。
如果你有写博客的好习惯
欢迎投稿
赞+在看,小生感恩❤️