当前位置: 首页 > news >正文

MTKAndroid12 解决SystemUI下拉框中,长按WIFI图标会导致崩溃问题

解决SystemUI下拉框中,长按WIFI图标会导致崩溃问题

文章目录

  • 场景
  • 参考资料
  • 修改文件
  • 解决方案
    • 日志
    • 源码分析
  • 总结


场景

在部分产品中偶发性发现,

  • SystemUI下拉框下拉后长按WIFI图标会导致崩溃问题,有时候是截屏、点击Home 按键后,长按WIFI图标崩溃。
  • 第一次刷完固件开机OK的,重启后会复现,或者直接长按 崩溃必现。

这个现象很奇怪,平常自己开发中并不是所有项目都会遇到,

参考资料

Android11 下拉菜单长按WIFI 图标SystemUI ANR
参考资料有部分完整的报错信息日志和源码分析,方便理解问题。 核心就是在Handler 里面创建了Handler,Android体系不允许的。

new Handler()和new Handler(Looper.getMainLooper())的区别是什么?

区别总结来说:

  • 主线程本身就有一个Lopper,在程序起来的时候就已经lopper() 了,所以在主线程里面创建Handler,直接 new
    Handler()。 这个handler 发送消息自动会在Lopper() 队列里面等待执行。

  • 子线程里面不一样了,你创建了一个new Handler(), 系统不允许就报错了。 因为本身这个Handler()
    创建了就是个死Handler,无法让消息转动起来的。 所以系统不允许就直接报错了,如果想这样就是要在子线程中创建Handler ,那么就两个方案: 让这个Handler 有自己的Looper,所有就有了Lopper.prepare 和 Looper.loop; 或者在这个主线程中穿件的Handler 和 主线程关联起来,构造方法传递一个MainLoop 不就行了嘛。

如果 必现,这样的解决方案是OK的,就是在创建Handler 前后加Lopper.prepare() 和 Lopper.loop() 让线程中的消息循环起来。
如果是偶现,这样的解决方式肯定是不合理的,下面会给出我的修改解决方案
如果需要深究偶发原因,还需进一步重点分析,问题原因很简单就是子线程里面跑了Handler.

修改文件

相关关联文件
/vendor/mediatek/proprietary/packages/apps/SystemUI/src/com/android/systemui/statusbar/phone/StatusBar.java
/vendor/mediatek/proprietary/packages/apps/SystemUI/src/com/android/systemui/assist/AssistManager.java


修改文件
/vendor/mediatek/proprietary/packages/apps/SystemUI/src/com/android/systemui/assist/AssistManager.java

具体修改内容
mAssistDisclosure = new AssistDisclosure(context, new Handler());

修改为如下:
mAssistDisclosure = new AssistDisclosure(context, new Handler(Looper.getMainLooper()));

		

在这里插入图片描述

解决方案

日志

这里直接张贴一下别人家的日志 下。


10-01 08:01:11.236  5792  5833 E AndroidRuntime: FATAL EXCEPTION: AsyncTask #1
10-01 08:01:11.236  5792  5833 E AndroidRuntime: Process: com.android.systemui, PID: 5792
10-01 08:01:11.236  5792  5833 E AndroidRuntime: java.lang.RuntimeException: Can't create handler inside thread Thread[AsyncTask #1,5,main] that has not called Looper.prepare()
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at android.os.Handler.<init>(Handler.java:227)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at android.os.Handler.<init>(Handler.java:129)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.assist.AssistManager.<init>(AssistManager.java:213)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.assist.AssistManager_Factory.provideInstance(AssistManager_Factory.java:107)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.assist.AssistManager_Factory.get(AssistManager_Factory.java:70)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.assist.AssistManager_Factory.get(AssistManager_Factory.java:17)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at dagger.internal.DoubleCheck.get(DoubleCheck.java:47)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.statusbar.phone.StatusBar.lambda$startActivityDismissingKeyguard$16(StatusBar.java:2709)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.statusbar.phone.StatusBar.lambda$startActivityDismissingKeyguard$16$StatusBar(Unknown Source:0)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at com.android.systemui.statusbar.phone.-$$Lambda$StatusBar$fPMIOsYMhFXVKHESAjUObpcgeJM.run(Unknown Source:10)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:305)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
10-01 08:01:11.236  5792  5833 E AndroidRuntime: 	at java.lang.Thread.run(Thread.java:923)

源码分析

根据报错日志,找到了源码

在这里插入图片描述
报错的地方,注意两个地方:

  • 报错的地方已经修改,Handler() 构造方法,传递一个Looper.getMainLooper() 放到主线程
  • 这个类上面有一个Inject 注解,这个注意下,在引用地方就不是new 对象了的。

可以根据实际日志,反推调用地方是哪里报错了,如下追踪到StatusBar.java 源码了。
在这里插入图片描述

通过get() 方法,获取对象,创建了AssistManager 对象, 但是它是放到子线程里面调用的。 然后在AssistManager 构造方法里面穿件了Handler,所以直接报错了。

总结

通过这里的笔记学习和相关资料参考

  • 搞清楚Handler、Loop 的基本原理 相关原理不复杂需要掌握基本知识;搞清楚Looper.loop Lopper.prepare、Loop.getMainLooper 这些方法的使用。

相关文章:

  • 深入了解Spring事务及其使用场景
  • go:前后端分离
  • (UI自动化测试)第二篇:元素定位的方法_name定位
  • 【学习】CMMM/DCMM等智能制造产业常见资质学习
  • hive:处理JSON格式(get_json_object()函数),连续登录题型,快速生成表(explode,stack),迭代计算
  • 使用flask_restful快速构建接口
  • AI生成移动端贪吃蛇游戏页面,手机浏览器打开即可玩
  • .net core集成MQTT服务端
  • Pytorch中的torch.utils.data.Dataset 类
  • Next-Auth 认证系统:用户与管理员双角色登录配置
  • 【深度技术揭秘】 Android SystemUI锁屏界面动态布局重构:横竖屏智能适配指南
  • 【最后203篇系列】022 用Deepseek14b提取新闻事件
  • 官方通知 | 2025年CAIP人工智能职场应用师(AI职场应用师)职业能力认证正式发布
  • 【机器学习】机器学习四大分类
  • Camera2 与 CameraX 闲谈
  • 【惯性系与固连系速度位置加速度转换关系】
  • Redis 内存淘汰策略
  • Compose 原理解析
  • 【信息系统项目管理师】【高分范文】【历年真题】​论信息系统项目的风险管理
  • 基于大模型的甲状舌管囊肿全流程预测与临床方案研究报告
  • 杭州发布最新“独角兽企业”榜单,“六小龙”中5家已晋级
  • 长三角与粤港澳大湾区融合发展,无锡何以成为窗口?
  • 建行原副行长章更生被开除党籍:靠贷吃贷,大搞权钱交易
  • 城事|喊侬白相,长兴太湖9号公路邀上海市民共赴诗意之旅
  • 重庆市委原常委、政法委原书记陆克华严重违纪违法被开除党籍和公职
  • 陈冬评价神二十乘组:合,三头六臂;分,独当一面