Android 中的多线程编程全面解析

Android 中的多线程编程全面解析

一、Android 线程模型基础

主线程(UI 线程)特性

  • 唯一性:每个应用只有一个主线程
  • 职责:处理 UI 操作和用户交互
  • 限制:禁止在主线程执行耗时操作(超过5秒会导致 ANR)
  • 检查方法Looper.getMainLooper().thread == Thread.currentThread()

工作线程使用原则

  • 网络请求:必须在工作线程执行
  • 文件IO:超过毫秒级的操作应放在工作线程
  • 数据库操作:Room 默认提供异步支持,复杂查询仍需工作线程
  • 计算密集型任务:如加密、图像处理等

二、Android 多线程方案演进

1. 基础方案

// 基本线程创建
Thread {// 后台操作runOnUiThread {// 更新UI}
}.start()

2. Executor 框架(推荐基础方案)

val executor = Executors.newFixedThreadPool(4)
executor.execute {// 后台任务
}

3. Android 特有机制

// HandlerThread 示例
val handlerThread = HandlerThread("MyHandlerThread").apply { start() }
val handler = Handler(handlerThread.looper)
handler.post {// 在后台线程执行
}

三、现代协程解决方案(Kotlin Coroutines)

协程核心概念

  • 轻量级线程:数千协程可并行,开销远小于线程
  • 结构化并发:自动取消和资源清理
  • 挂起函数:用同步写法实现异步操作

基本使用

// ViewModel 中启动协程
viewModelScope.launch {// 主线程执行val data = withContext(Dispatchers.IO) {// 切换到IO线程执行网络请求fetchDataFromNetwork()}// 自动切回主线程更新UIupdateUI(data)
}

关键组件

组件作用典型使用场景
Dispatchers.Main主线程更新UI
Dispatchers.IOIO密集型网络/文件操作
Dispatchers.DefaultCPU密集型复杂计算
viewModelScope自动取消作用域ViewModel中的协程
lifecycleScope生命周期作用域Activity/Fragment中的协程

四、线程间通信机制

1. Handler/Looper 传统方式

val mainHandler = Handler(Looper.getMainLooper())
Thread {// 后台工作mainHandler.post {// 主线程更新UI}
}.start()

2. LiveData 自动线程切换

liveData.postValue(data) // 可从任何线程调用
liveData.observe(this) { data -> // 自动在主线程回调
}

3. Flow 响应式流

flow {emit(fetchData()) // 在IO线程发射数据
}
.map { compute(it) }  // 在Default线程转换
.flowOn(Dispatchers.Default)
.collect { data ->    // 在主线程收集updateUI(data)
}

五、高级并发模式

1. 并发任务处理

// 并行执行多个请求
val deferred1 = async { fetchData1() }
val deferred2 = async { fetchData2() }
val result = deferred1.await() + deferred2.await()

2. 互斥与同步

val mutex = Mutex()
var sharedData = 0fun updateData() = runBlocking {mutex.withLock {sharedData++}
}

3. 超时控制

withTimeoutOrNull(3000) {fetchData() // 超过3秒自动取消
} ?: showTimeoutError()

六、性能优化与陷阱规避

最佳实践

  1. 线程池配置

    // 根据CPU核心数优化线程池
    val cpuCount = Runtime.getRuntime().availableProcessors()
    val executor = ThreadPoolExecutor(cpuCount + 1,cpuCount * 2 + 1,30L, TimeUnit.SECONDS,LinkedBlockingQueue()
    )
    
  2. 避免内存泄漏

    lifecycleScope.launch {lifecycle.repeatOnLifecycle(Lifecycle.State.STARTED) {flow.collect { data ->// 只在界面可见时处理数据}}
    }
    
  3. ANR 预防

    • 将超过100ms的操作移出主线程
    • 使用 StrictMode 检测违规操作
    StrictMode.setThreadPolicy(StrictMode.ThreadPolicy.Builder().detectNetwork().penaltyLog().build())
    

常见陷阱

  1. 回调地狱

    // 错误示例
    fetchUser { user ->fetchProfile(user.id) { profile ->updateUI(user, profile) // 嵌套难以维护}
    }// 协程改造
    viewModelScope.launch {val user = fetchUserSuspend()val profile = fetchProfileSuspend(user.id)updateUI(user, profile)
    }
    
  2. 过度创建线程

    • 避免直接 new Thread(),应使用线程池
    • 协程不应执行阻塞操作(应用 withContext
  3. 并发修改异常

    // 错误示例
    val list = mutableListOf<Int>()
    (1..1000).forEach { i ->thread {list.add(i) // 可能崩溃}
    }// 正确方案
    val concurrentList = Collections.synchronizedList(mutableListOf<Int>())
    

七、调试与监控工具

  1. Android Studio Profiler

    • 查看线程状态和CPU使用率
    • 检测线程泄漏和阻塞
  2. StrictMode

    StrictMode.setVmPolicy(VmPolicy.Builder().detectLeakedClosableObjects().penaltyLog().build())
    
  3. 协程调试

    // 在调试模式下启用协程名称
    System.setProperty("kotlinx.coroutines.debug", "on")
    
  4. 自定义线程监控

    // 监控线程创建
    val threadFactory = ThreadFactory { r ->Thread(r).also { thread ->Log.d("ThreadMonitor", "Thread created: ${thread.name}")}
    }
    

Android 多线程编程已经从原始的 Thread+Handler 模式发展为以协程为核心的现代并发模型。开发者应当根据项目需求选择合适方案,小型项目可使用 Executor + Handler,中大型项目推荐全面采用协程。无论采用何种方案,都需要注意线程安全、生命周期管理和性能优化,才能构建出响应迅速且稳定的 Android 应用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/914251.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/914251.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

golang -gorm 增删改查操作,事务操作

增删改查 1. 插入数据// api func SaveUser(ctx *gin.Context) {result : &common.Result{}user : &dao.User{}err : ctx.ShouldBindJSON(&user)if err ! nil {ctx.JSON(http.StatusOK, result.Fail(400, "请使用json数据格式传值"))return}// 调用验证函…

大数据时代UI前端的智能化服务升级:基于用户情境的主动服务设计

hello宝子们...我们是艾斯视觉擅长ui设计、前端开发、数字孪生、大数据、三维建模、三维动画10年经验!希望我的分享能帮助到您!如需帮助可以评论关注私信我们一起探讨!致敬感谢感恩!一、引言&#xff1a;从 “被动响应” 到 “主动预判” 的 UI 服务革命当用户在暴雨天打开外卖…

CUDA性能优化实战:7个步骤让并行归约算法提升10倍效率

本文深入探讨了一个经典的并行计算算法——并行归约&#xff08;Parallel Reduction&#xff09;的性能优化过程&#xff0c;通过七个渐进式的优化步骤&#xff0c;展示了如何将算法性能提升至极致。这项研究基于Mark Harris在NVIDIA网络研讨会中提出的优化方法&#xff0c;在重…

详解梯度消失和梯度爆炸(反向传播)?

什么是梯度消失&#xff1f;梯度消失&#xff08;Gradient Vanishing&#xff09; 是指在训练神经网络时&#xff0c;反向传播过程中计算得到的梯度&#xff08;用于更新参数的重要信息&#xff09;随着网络层数的增加而急剧减小&#xff0c;甚至趋近于零的现象。这会导致深层网…

端到端自动驾驶:挑战与前沿

端到端自动驾驶&#xff1a;挑战与前沿 End-to-End Autonomous Driving: Challenges and Frontiers 自动驾驶研究社区已见证了越来越多采用端到端算法框架的方法的快速增长&#xff0c;这些方法利用原始传感器输入生成车辆的运动规划&#xff0c;而不是专注于诸如检测和运动预测…

rust cargo 编译双架构的库

这个错误表明你的 Rust 工具链没有安装 aarch64-apple-darwin 目标平台。以下是完整的解决方案&#xff1a; 解决方案 ​​安装目标平台​​ (必须步骤) rustup target add aarch64-apple-darwin​​验证安装​​ (可选但推荐) rustup target list --installed # 应该能看到 aa…

Apache Shiro 框架详解

文章目录一、Shiro 核心功能二、Shiro 架构2.1 三层架构2.2 核心组件&#xff08;SecurityManager 内部&#xff09;三、核心流程详解3.1 认证流程&#xff08;登录&#xff09;流程步骤&#xff1a;认证流程序列图&#xff1a;3.2 授权流程&#xff08;权限校验&#xff09;流…

【保姆级喂饭教程】Windows下安装Git Flow

目录前言一、SourceTree二、Git for Windows (previously MSysGit)1. 下载补丁1.1 getopt.exe1.2 libintl3.dll1.3 libiconv2.dll1.4 安装补丁2. 安装Git Flow3. 测试3.1 初始化&#xff08;Initialize&#xff09;3.2 设置远程3.3 创建分支3.4 功能开发3.5 功能提交3.6 推送分…

manifest.json只有源码视图没其他配置

项目场景&#xff1a;提示&#xff1a;这里简述项目相关背景&#xff1a;有时候我们从git上面拉下代码&#xff0c;第一次运行时发现&#xff0c;没运行项&#xff0c;再看manifest.json文件&#xff0c;就只有json文件&#xff0c;没有其他配置项原因分析&#xff1a;提示&…

数据分析-名词

一、网页访问数据指标1.IP &#xff08;Internet Protocol&#xff09;独立IP 通常采用独立IP数&#xff0c; 理论上指00:00-24:00内相同IP地址重复访问只被计算一次。而不同的商业统计工具&#xff0c;缩短去 掉重复统计的时间&#xff0c;也是数据统计放大的一个常用套路。 &…

UDP属于是一种什么服务器?

UDP是一种传输层协议&#xff0c;通常会被应用在计算机网络中&#xff0c;为企业与用户提供无连接的数据信息传输功能&#xff0c;与TCP协议相比较来说&#xff0c;UDP会更加的简单但是UDP在可靠性方面没有一定的保证&#xff0c;属于是一种基于UDP协议进行通信的服务器。UDP服…

ARM单片机OTA解析(一)

文章目录一、单片机烧写程序的几种方法二、Bootloader如何加载启动App一、单片机烧写程序的几种方法 在线应用编程&#xff0c;由开发者实现Bootloader功能&#xff0c;比如ARM单片机的Code分区中的Flash本是存储用户应用程序的区间(上电从此处执行用户代码)&#xff0c;开发者…

C语言基础教程--从入门到精通

C语言基础教程–从入门到精通&#xff08;总体概括&#xff09; 接下来会对每一个章节进行详细的总结与整理&#xff0c;希望对大家有用&#xff01;大家一起学习&#xff01; 目录C语言基础教程--从入门到精通&#xff08;总体概括&#xff09;**接下来会对每一个章节进行详细…

单细胞分析教程 | (二)标准化、特征选择、降为、聚类及可视化

在完成质控&#xff08;QC&#xff09;后&#xff0c;我们已经过滤掉了低质量细胞、双细胞和低表达基因&#xff0c;获得了较为干净的单细胞数据集单细胞分析教程 | &#xff08;一&#xff09;Python单细胞质控全流程。接下来&#xff0c;我们将进行以下关键步骤&#xff1a; …

大模型 Agent(智能体)技术简介

大模型 Agent&#xff08;智能体&#xff09;技术 是当前人工智能领域的前沿方向&#xff0c;它赋予大型语言模型&#xff08;LLM&#xff09;自主感知、规划、决策和行动的能力&#xff0c;使其不再局限于“被动应答”&#xff0c;而是能主动完成复杂任务。简单来说&#xff0…

OneCode 3.0架构深度剖析:工程化模块管理与自治UI系统的设计与实现

引言 OneCode 3.0作为新一代低代码开发平台&#xff0c;其架构设计围绕"工程模块化"与"UI自主化"两大核心目标展开。本文将从底层接口到上层应用&#xff0c;全面解析OneCode 3.0的技术架构&#xff0c;包括核心工厂类、工程管理接口、数据仓库设计以及动态…

功耗校准数据PowerProfile测试方法建议

场景步骤版本:xxxxA1A2结果&#xff08;mA&#xff09;screen,full1.打开飞行模式&#xff0c;灭屏时间最长&#xff0c;其他的基础功能关2.进入到日历应用界面3.将亮度设置至最大&#xff08;4095&#xff09;&#xff0c;待电流稳定后&#xff0c;测试5分钟&#xff0c;记录电…

[附源码+数据库+毕业论文]基于Spring+MyBatis+MySQL+Maven+vue实现的供电公司安全生产考试管理系统,推荐!

摘 要 使用旧方法对安全生产考试信息进行系统化管理已经不再让人们信赖了&#xff0c;把现在的网络信息技术运用在安全生产考试信息的管理上面可以解决许多信息管理上面的难题&#xff0c;比如处理数据时间很长&#xff0c;数据存在错误不能及时纠正等问题。 这次开发的供电公…

输入框过滤选项列表,el-checkbox-group单选

需求&#xff1a;根据输入的文本动态过滤选项列表&#xff0c;并在下方显示匹配到的选项。当用户勾选匹配到的选项时&#xff0c;把该选项的值赋值给输入框中绑定的值。当用户取消选择时&#xff0c;输入框中的字段可以随意编辑。组件&#xff1a;el-input、el-checkbox-group、…

身份认证缺陷

Authentication Bypasses审计创建AccountVerificationHelper实例&#xff0c;用于处理账户验证逻辑parseSecQuestions函数的作用是从请求体中遍历参数名&#xff0c;找到包含secQuestion的参数&#xff0c;将其值存入Map中并返回这里直接把AccountVerificationHelper整个分析一…