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

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 这些方法的使用。

http://www.kler.cn/a/598880.html

相关文章:

  • 3.20-1ui自动化切换,登录退出
  • 使用 Docker 构建 LangChain 开发环镜及 ChatOllama 示例
  • [vue]属性绑定
  • python总结
  • Pytorch系列教程:微调BERT实现命名实体识别
  • SpringBoot与Redisson整合,用注解方式解决分布式锁的使用问题
  • 查找单入口空闲区域[A卷-hw_od]
  • JS Typeof 运算符
  • 系统设计类问题回答模板
  • SQL Server行转列操作及PIVOT运算符
  • 算法基础篇(1)(蓝桥杯常考点)
  • FPGA时钟约束
  • 基于Matlab实现语音识别算法(源码+数据)
  • 【Linux文件IO】通过文件IO把bmp图片显示到Linux开发板的实现
  • 基于springboot的新闻推荐系统(045)
  • 【NLP 42、实践 ⑪ 用Bert模型结构实现自回归语言模型的训练】
  • 人脸表情识别系统分享(基于深度学习+OpenCV+PyQt5)
  • Spring Boot框架中常用注解
  • 【mysql】同一个字段,字符串相加
  • 从Oracle到OceanBase数据库迁移:全方位技术解析