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

子线程不能直接 new Handler(),而主线程可以

在 Android 中,子线程不能直接 new Handler(),而主线程可以,原因在于 Looper 机制。下面详细解释:


1. 为什么主线程可以直接 new Handler()

主线程(UI 线程)在启动时,系统会自动调用 Looper.prepareMainLooper()Looper.loop(),为主线程初始化一个 Looper 并启动消息循环。因此,在主线程中:

Handler handler = new Handler(); // 直接创建,默认绑定主线程的 Looper

等价于:

Handler handler = new Handler(Looper.getMainLooper()); // 显式指定主线程 Looper

2. 为什么子线程不能直接 new Handler()

子线程默认没有初始化 Looper,直接 new Handler() 会抛出异常:

// 子线程中直接调用会崩溃!
new Handler(); // 抛出 RuntimeException: "Can't create handler inside thread that has not called Looper.prepare()"
原因:
  • Handler 需要绑定一个 Looper 来管理消息队列(MessageQueue)。
  • 子线程的 Looper 需要手动初始化,否则 Handler 无法找到可用的 Looper

3. 如何在子线程正确创建 Handler

必须显式调用 Looper.prepare()Looper.loop()

new Thread(() -> {// 1. 初始化 LooperLooper.prepare(); // 2. 创建 Handler(此时会绑定当前线程的 Looper)Handler handler = new Handler(); // 3. 启动消息循环(必需!否则 Handler 无法处理消息)Looper.loop(); 
}).start();
注意事项:
  • 如果子线程的 Handler 需要更新 UI,必须通过 runOnUiThread 或主线程 Handler 转发。
  • 退出子线程时,需调用 Looper.myLooper().quit() 释放资源,否则可能导致内存泄漏。

4. 为什么 Android 这样设计?

  • 主线程:需要处理 UI 事件(如触摸、绘制),必须有一个常驻的消息循环(Looper),因此系统自动初始化。
  • 子线程:通常是临时执行任务,默认不维护消息循环,避免不必要的性能开销。如果需要异步消息机制(如 HandlerThread),再手动初始化 Looper

5. 简化子线程 Handler 的写法

Android 提供了 HandlerThread 类,封装了 Looper 的创建和销毁:

// 创建带 Looper 的子线程
HandlerThread handlerThread = new HandlerThread("MyHandlerThread");
handlerThread.start();// 获取子线程的 Looper 创建 Handler
Handler handler = new Handler(handlerThread.getLooper());// 退出时释放资源
handlerThread.quit();

总结

场景能否直接 new Handler()原因
主线程✅ 可以系统自动初始化 Looper
子线程❌ 不能默认无 Looper,需手动调用 Looper.prepare()

关键点

  • Handler 必须绑定一个 Looper,而 Looper 需要消息循环(Looper.loop())才能工作。
  • 子线程若需使用 Handler,需按规范初始化 Looper,或直接使用 HandlerThread
http://www.xdnf.cn/news/1151119.html

相关文章:

  • sql练习二
  • 打靶日记之xss-labs
  • OpenCV 官翻 4 - 相机标定与三维重建
  • 如何设计一个软件项目管理系统:架构设计合集(六)
  • 小明记账簿焕新记:从单色到多彩的主题进化之路
  • Java并发8--并发安全容器详解
  • Springboot项目的搭建方式5种
  • Tomcat 生产 40 条军规:容量规划、调优、故障演练与安全加固
  • day25 力扣90.子集II 力扣46.全排列 力扣47.全排列 II
  • LVS(Linux virual server)
  • windows内核研究(驱动开发-0环与3环的通信)
  • Kotlin泛型约束
  • 多表查询-8-练习总结
  • 数据库练习3
  • Flowable31动态表单-----------------------终章
  • 博图SCL语言中常用运算符使用详解及实战案例(下)
  • OpenCV 官翻 3 - 特征检测 Feature Detection
  • 【无标题】重点阅读——如何在信息层面区分和表征卷曲维度,解析黑洞内部的维度区分机制
  • 《命令行参数与环境变量:从使用到原理的全方位解析》
  • 搭建比分网服务器怎么选数据不会卡顿?
  • lvs原理及实战部署
  • 【I2C】01.I2C硬件连接I2C总线时序图讲解
  • Go语言pprof性能分析指南
  • Temperature 是在LLM中的每一层发挥作用,还是最后一层? LLM中的 Temperature 参数 是怎么计算的
  • 操作系统-分布式同步
  • TCP/UDP协议深度解析(四):TCP的粘包问题以及异常情况处理
  • GaussDB 数据库架构师修炼(六) 集群工具管理-1
  • 异步解决一切问题 |消息队列 |减少嵌套 |hadoop |rabbitmq |postsql
  • 深入解析 Amazon Q:AWS 推出的企业级生成式 AI 助手
  • 【设计模式C#】外观模式(用于解决客户端对系统的许多类进行频繁沟通)