Kotlin 协程:深入理解 ‘lifecycleScope‘

Kotlin 协程:深入理解 ‘lifecycleScope’

在这里插入图片描述

Kotlin 协程是一种强大的异步编程工具,它提供了一种简洁、易读的方式来处理并发和异步操作。在 Kotlin 协程库中,lifecycleScope 是一个关键的概念,它允许我们将协程的生命周期绑定到 Android 组件的生命周期。在本篇博客中,我们将深入探讨 lifecycleScope 的工作原理,以及如何在实际的 Kotlin 代码中使用它。

协程简介

在我们深入探讨 lifecycleScope 之前,让我们先简单回顾一下协程的基本概念。协程是一种可以挂起和恢复执行的计算。与线程不同,协程的挂起和恢复不需要操作系统的介入,因此协程的开销非常小。这使得我们可以在一个程序中同时运行大量的协程,而不会像线程那样消耗大量的系统资源。

Kotlin 提供了一套丰富的协程 API,使得我们可以轻松地在 Kotlin 程序中使用协程。这套 API 的核心是 suspend 函数和 CoroutineScopesuspend 函数是一种可以被挂起和恢复的函数,它可以在不阻塞线程的情况下执行长时间运行的操作,如网络请求或数据库查询。CoroutineScope 是一种定义协程生命周期的范围,它提供了一种方式来启动新的协程,并管理它们的生命周期。

在 Android 开发中,我们通常会使用 lifecycleScopeviewModelScope 这样的预定义范围来启动和管理协程。这些范围将协程的生命周期绑定到 Android 组件的生命周期,使得我们可以在组件的生命周期内安全地使用协程,而不用担心内存泄漏或者应用崩溃。

什么是 lifecycleScope

lifecycleScope 是一个绑定到 Android 组件生命周期的 CoroutineScope。当组件(如 Activity 或 Fragment)被销毁时,lifecycleScope 会自动取消它启动的所有协程。这使得我们可以在组件的生命周期内安全地使用协程,而不用担心内存泄漏或者应用崩溃。

lifecycleScope 是通过 LifecycleCoroutineScope 类实现的,这个类是 CoroutineScope 的一个子类。LifecycleCoroutineScope 有一个和组件生命周期相关联的 Lifecycle 对象,当这个 Lifecycle 对象的状态变为 DESTROYED 时,LifecycleCoroutineScope 会自动取消它的所有协程。

我们可以通过 lifecycleScope 属性来获取一个组件的 lifecycleScope。这个属性是在 LifecycleOwner 接口中定义的,所有的 Android 组件,如 Activity 和 Fragment,都实现了这个接口。所以,我们可以在任何一个组件中直接使用 lifecycleScope

class MyActivity : AppCompatActivity() {fun loadData() {lifecycleScope.launch {// 在这里执行异步操作}}
}

在这个例子中,我们在 MyActivity 中使用 lifecycleScope 启动了一个新的协程。这个协程的生命周期会被绑定到 MyActivity 的生命周期,当 MyActivity 被销毁时,这个协程也会被自动取消。

lifecycleScope vs GlobalScope

在 Kotlin 协程库中,除了 lifecycleScope,我们还可以使用 GlobalScope 来启动协程。GlobalScope 是一个全局的 CoroutineScope,它的生命周期是整个应用程序的生命周期。当我们在 GlobalScope 中启动一个协程时,这个协程会一直运行,直到它的工作完成,或者应用程序被销毁。

尽管 GlobalScope 看起来很方便,但在实际的 Android 开发中,我们通常不推荐使用它。因为 GlobalScope 的生命周期过长,如果我们在 GlobalScope 中启动了一个长时间运行的协程,那么这个协程可能会在它的工作完成之前一直占用系统资源,这可能会导致内存泄漏或者应用崩溃。

相比之下,lifecycleScope 的生命周期更短,它只会在组件的生命周期内运行。这使得我们可以更安全地使用协程,而不用担心内存泄漏或者应用崩溃。

使用 lifecycleScope 启动协程

lifecycleScope 中启动协程非常简单。我们只需要调用 lifecycleScopelaunchasync 方法,然后在 {} 中编写我们的异步代码即可。

class MyActivity : AppCompatActivity() {fun loadData() {lifecycleScope.launch {// 在这里执行异步操作val data = fetchDataFromNetwork()displayData(data)}}
}

在这个例子中,我们在 MyActivitylifecycleScope 中启动了一个新的协程。这个协程首先从网络中获取数据,然后将这些数据显示到界面上。由于我们是在 lifecycleScope 中启动的这个协程,所以当 MyActivity 被销毁时,这个协程也会被自动取消。

lifecycleScopesuspend 函数

lifecycleScopesuspend 函数是 Kotlin 协程的两个核心概念,它们经常一起使用。在 lifecycleScope 中,我们可以调用 suspend 函数来执行长时间运行的操作,如网络请求或数据库查询。由于 suspend 函数可以被挂起和恢复,所以它们不会阻塞主线程,这使得我们可以在主线程中安全地使用 suspend 函数。

class MyActivity : AppCompatActivity() {fun loadData() {lifecycleScope.launch {// 在这里调用 suspend 函数val data = fetchDataFromNetwork()displayData(data)}}suspend fun fetchDataFromNetwork(): Data {// 在这里执行网络请求...}
}

在这个例子中,我们在 MyActivitylifecycleScope 中启动了一个新的协程。这个协程首先调用 fetchDataFromNetwork 函数从网络中获取数据,然后将这些数据显示到界面上。由于 fetchDataFromNetwork 是一个 suspend 函数,所以它可以在不阻塞主线程的情况下执行网络请求。

lifecycleScopeDispatchers

在 Kotlin 协程中,Dispatchers 是一个重要的概念,它决定了协程应该在哪个线程上执行。Dispatchers 有三个预定义的实例:Dispatchers.MainDispatchers.IODispatchers.DefaultDispatchers.Main 用于在主线程上执行协程,Dispatchers.IO 用于在 IO 线程上执行协程,Dispatchers.Default 用于在默认线程上执行协程。

当我们在 lifecycleScope 中启动协程时,我们可以通过 withContext 函数来改变协程的调度器。这使得我们可以在 lifecycleScope 中启动的协程中,根据需要在不同的线程上执行不同的操作。

class MyActivity : AppCompatActivity() {fun loadData() {lifecycleScope.launch {// 在主线程中更新 UIshowLoadingIndicator()// 在 IO 线程中执行网络请求val data = withContext(Dispatchers.IO) {fetchDataFromNetwork()}// 在主线程中更新 UIdisplayData(data)hideLoadingIndicator()}}suspend fun fetchDataFromNetwork(): Data {// 在这里执行网络请求...}
}

在这个例子中,我们首先在主线程中显示加载指示器,然后在 IO 线程中执行网络请求,最后再在主线程中显示数据和隐藏加载指示器。这是通过 withContext 函数和 Dispatchers.IO 实现的,它们让我们可以在 lifecycleScope 中启动的协程中,根据需要在不同的线程上执行不同的操作。

lifecycleScopeviewModelScope

在 Android 开发中,除了 lifecycleScope,我们还常常使用 viewModelScopeviewModelScope 是一个绑定到 ViewModel 生命周期的 CoroutineScope。当 ViewModel 被清除时,viewModelScope 会自动取消它启动的所有协程。

viewModelScopelifecycleScope 的主要区别在于它们的生命周期:lifecycleScope 的生命周期是组件的生命周期,而 viewModelScope 的生命周期是 ViewModel 的生命周期。通常,ViewModel 的生命周期比组件的生命周期要长,因为 ViewModel 会在配置更改(如屏幕旋转)时保持存活,而组件则会在配置更改时被销毁和重新创建。

所以,如果我们需要在配置更改后继续执行的协程,我们应该在 viewModelScope 中启动这个协程。如果我们的协程只需要在组件的生命周期内运行,那么我们应该在 lifecycleScope 中启动这个协程。

class MyViewModel : ViewModel() {fun loadData() {viewModelScope.launch {// 在这里执行异步操作val data = fetchDataFromNetwork()displayData(data)}}
}class MyActivity : AppCompatActivity() {private val viewModel: MyViewModel by viewModels()override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)viewModel.loadData()}
}

在这个例子中,我们在 MyViewModelviewModelScope 中启动了一个新的协程。这个协程的生命周期会被绑定到 MyViewModel 的生命周期,当 MyViewModel 被清除时,这个协程也会被自动取消。

lifecycleScopelaunchWhenCreatedlaunchWhenStartedlaunchWhenResumed

lifecycleScope 中,我们可以使用 launchWhenCreatedlaunchWhenStartedlaunchWhenResumed 这几个函数来在特定的生命周期状态下启动协程。这些函数会在组件达到指定的生命周期状态时启动协程,并在组件的生命周期状态变为 DESTROYED 时取消协程。

launchWhenCreatedlaunchWhenStartedlaunchWhenResumed 的主要区别在于它们启动协程的时机:launchWhenCreated 会在组件的生命周期状态至少为 CREATED 时启动协程,launchWhenStarted 会在组件的生命周期状态至少为 STARTED 时启动协程,launchWhenResumed 会在组件的生命周期状态至少为 RESUMED 时启动协程。

class MyActivity : AppCompatActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)lifecycleScope.launchWhenCreated {// 在这里执行异步操作val data = fetchDataFromNetwork()displayData(data)}}
}

在这个例子中,我们在 MyActivitylifecycleScope 中使用 launchWhenCreated 启动了一个新的协程。这个协程会在 MyActivity 的生命周期状态至少为 CREATED 时开始执行,当 MyActivity 被销毁时,这个协程也会被自动取消。

lifecycleScopeJobDeferred

在 Kotlin 协程中,JobDeferred 是两个重要的概念,它们代表了协程的工作。Job 是一个可以被取消的工作,Deferred 是一个可以获取结果的工作。我们可以通过 JobDeferred 来控制协程的执行,如取消协程、获取协程的结果、等待协程完成等。

当我们在 lifecycleScope 中启动协程时,launchasync 方法会返回一个 JobDeferred 对象。我们可以保存这个对象,然后在需要的时候用它来控制协程的执行。

class MyActivity : AppCompatActivity() {private var job: Job? = nullfun loadData() {job = lifecycleScope.launch {// 在这里执行异步操作val data = fetchDataFromNetwork()displayData(data)}}fun cancelDataLoading() {job?.cancel()}
}

在这个例子中,我们在 MyActivitylifecycleScope 中启动了一个新的协程,并将 launch 方法返回的 Job 对象保存在 job 变量中。然后,我们可以在 cancelDataLoading 方法中使用 job 对象来取消这个协程。

lifecycleScopeCoroutineExceptionHandler

在 Kotlin 协程中,我们可以使用 CoroutineExceptionHandler 来处理协程中的未捕获异常。CoroutineExceptionHandler 是一个特殊的上下文元素,它定义了当协程中的异常未被捕获时应该做什么。

当我们在 lifecycleScope 中启动协程时,我们可以通过 CoroutineExceptionHandler 来处理协程中的未捕获异常。我们只需要创建一个 CoroutineExceptionHandler 对象,然后在启动协程时将它作为上下文参数传入即可。

class MyActivity : AppCompatActivity() {private val exceptionHandler = CoroutineExceptionHandler { _, exception ->// 在这里处理异常displayError(exception)}fun loadData() {lifecycleScope.launch(exceptionHandler) {// 在这里执行异步操作val data = fetchDataFromNetwork()displayData(data)}}
}

在这个例子中,我们首先创建 一个 CoroutineExceptionHandler 对象,并在 launch 方法中将它作为一个参数。这样,如果在协程中出现未捕获的异常,CoroutineExceptionHandler 将会处理这个异常,我们可以在里面执行错误处理的代码。

lifecycleScopeSupervisorJob

在 Kotlin 协程中,SupervisorJobJob 的一个特殊子类,它具有一种特性:如果一个子协程失败了,它不会取消它的兄弟协程。这使得我们可以在一个 SupervisorJob 中启动多个协程,而不用担心一个协程的失败会影响其他协程。

当我们在 lifecycleScope 中启动协程时,我们可以使用 SupervisorJob 来控制协程的执行。我们只需要创建一个 SupervisorJob 对象,然后在启动协程时将它作为上下文参数传入即可。

class MyActivity : AppCompatActivity() {private val supervisorJob = SupervisorJob()fun loadData() {lifecycleScope.launch(supervisorJob) {// 在这里执行异步操作val data = fetchDataFromNetwork()displayData(data)}}
}

在这个例子中,我们首先创建了一个 SupervisorJob 对象,并在 launch 方法中将它作为一个参数。这样,即使在协程中出现未捕获的异常,这个协程的兄弟协程仍然会继续执行。

lifecycleScopeFlow

在 Kotlin 协程中,Flow 是一个表示异步数据流的概念。我们可以使用 Flow 来表示一系列的异步事件,如网络请求的结果、数据库查询的结果、用户输入的事件等。

当我们在 lifecycleScope 中启动协程时,我们可以使用 Flow 来处理异步的数据流。我们只需要创建一个 Flow 对象,然后在协程中调用 collect 方法来收集 Flow 的数据。

class MyActivity : AppCompatActivity() {fun loadData() {lifecycleScope.launch {fetchDataFromNetwork().flowOn(Dispatchers.IO).collect { data ->// 在主线程中更新 UIdisplayData(data)}}}fun fetchDataFromNetwork(): Flow<Data> {// 在这里返回一个 Flow 对象...}
}

在这个例子中,我们首先创建了一个 Flow 对象,然后在 lifecycleScope 中启动的协程中收集这个 Flow 的数据。由于 Flow 是异步的,所以我们可以在不阻塞主线程的情况下收集 Flow 的数据。

结论

lifecycleScope 是 Kotlin 协程库中的一个强大工具,它允许我们将协程的生命周期绑定到 Android 组件的生命周期。通过使用 lifecycleScope,我们可以在 Android 应用中安全地使用协程,而不用担心内存泄漏或应用崩溃。

在这篇博客中,我们介绍了 lifecycleScope 的基本概念和使用方法,以及如何使用它来处理复杂的并发场景和优化我们的代码。我希望这些信息对你有所帮助,如果你有任何问题或建议,欢迎在评论区留言。

保持编程,保持学习!

感谢阅读, Best Regards!

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

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

相关文章

第二十一回 阎婆大闹郓城县 朱仝义释宋公明-FreeBSD Linux 使用Rsync备份

阎婆状告宋江杀死她女儿阎婆惜&#xff0c;知县有意偏袒宋江&#xff0c;只是一味的拷打唐牛儿&#xff0c;但无奈张三张文远说刀子是宋江的&#xff0c;知县不得已差人拿宋江来审问。第一次没见到人&#xff0c;第二次派朱仝雷横两个人去。 朱仝到地窖里找到了躲藏的宋江&…

QT使用QFileSystemModel实现的文件资源管理器(开源)

文章目录 效果图现实的功能总体框架功能介绍视图双击进入处理复制与剪切粘贴重命名&#xff0c;新建显示文件详细信息文件路径导航栏 总结 效果图 现实的功能 支持文件/文件夹复制&#xff0c;粘贴&#xff0c;剪切&#xff0c;删除&#xff0c;重命名的基本操作支持打开图片&…

【算法分析与设计】交换两个节点

&#x1f4dd;个人主页&#xff1a;五敷有你 &#x1f525;系列专栏&#xff1a;算法分析与设计 ⛺️稳中求进&#xff0c;晒太阳 题目 给你一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后链表的头节点。你必须在不修改节点内部的值的情况下完成本…

Python程序设计 函数基础

简单函数 函数&#xff1a;就是封装了一段可被重复调用执行的代码块。通过此代码块可以实现大量代码的重复使用。 函数的使用包含两个步骤&#xff1a; 定义函数 —— 封装 独立的功能 调用函数 —— 享受 封装 的成果 函数的作用&#xff0c;在开发程序时&#xff0c;使用…

【毕业日记】2024.01 - 慢下来,静待花开

转眼距离930离开鹅厂已经120天了&#xff0c;我是很能拖延的&#xff0c;或者是很懂自我麻痹的&#xff0c;这三个多月&#xff0c;一直想要写点东西纪念&#xff0c;一直拖一直拖一直拖…… 疫情这几年经济下行里裁员是个茶余饭后“嬉笑”之余经常被提起的词&#xff0c;部门滚…

【Qt基本功修炼】Qt线程的两种运行模式

1. 前言 QThread是Qt中的线程类&#xff0c;用于实现多线程运行。 QThread有两种工作模式&#xff0c;即 消息循环模式无消息循环模式 两种模式分别适用于不同的场景。下面我们将从多个方面&#xff0c;讲解QThread两种工作模式的区别。 2. 消息循环模式 2.1 实现原理 Q…

PAT-Apat甲级题1004(python和c++实现)

PTA | 1004 Counting Leaves 1004 Counting Leaves 作者 CHEN, Yue 单位 浙江大学 A family hierarchy is usually presented by a pedigree tree. Your job is to count those family members who have no child. Input Specification: Each input file contains one te…

排序(5)——归并排序

六、归并排序 1.简介 归并排序也是一种很经典的排序算法&#xff0c;采用分治的思想方法进行数据的处理。归并讲究的是先拆后合&#xff0c;也就是分治中的分而治之。在拿到一组数据后&#xff0c;程序会将整个数据进行不断拆分直至有序&#xff0c;因为单个元素必然有序&…

教你怎么给接口定义错误码

错误码一般包括三个要素&#xff1a; 前缀标识&#xff1a;区分错误类型&#xff0c;比如请求不合法&#xff0c;还是服务器处理错误。模块标识&#xff1a;区分到底是哪个模块的错误。错误代码&#xff1a;区分具体是什么问题。 从错误码的符号组成上看又分为三类&#xff1…

网络发展历程及SD-WAN的优势

自网络出现以来&#xff0c;经历了不同时期的发展阶段&#xff0c;而SD-WAN作为网络发展的新趋势&#xff0c;以其独特的优势满足了当今企业对于高效、灵活和成本控制的网络需求。 一、网络发展历程 &#xff08;1&#xff09;初始阶段&#xff1a;专线网络 从1980年代到2000…

VBA数据库解决方案第八讲:SQL语句及打开记录集

《VBA数据库解决方案》教程&#xff08;版权10090845&#xff09;是我推出的第二套教程&#xff0c;目前已经是第二版修订了。这套教程定位于中级&#xff0c;是学完字典后的另一个专题讲解。数据库是数据处理的利器&#xff0c;教程中详细介绍了利用ADO连接ACCDB和EXCEL的方法…

flowable 设置自定义属性教程

概述 由于工作需要给flowable工作流设计器添加自定义属性&#xff0c;以满足功能实现。所以这篇文章介绍下用flowable 开源的的flowable-ui 前端添加自定义属性&#xff0c;后端解析属性值的例子。 技术栈 序号技术点名称版本1Flowable6.8.0 使用的是flowable6.8.0 版的代码…

画质和场景双需求下,海信电视U8KL的变与不变

又到一年春节&#xff0c;最近几年大家过年的方式越来越丰富&#xff0c;但是跟家人在一起看春晚依然是主流&#xff0c;电视也是过年不可或缺的家庭成员。 当然&#xff0c;随着大家对生活品质的要求更高&#xff0c;对电视的要求也变得更高了。比如&#xff0c;现在春晚直播…

爬虫笔记(三):实战qq登录

咳咳&#xff0c;再这样下去会进橘子叭hhhhhh 以及&#xff0c;这个我觉得大概率是成功的&#xff0c;因为测试了太多次&#xff0c;登录并且验证之后&#xff0c;qq提醒我要我修改密码才可以登录捏QAQ 1. selenium 有关selenium具体是啥&#xff0c;这里就不再赘述了&#x…

leetcode c++ 超出内存限制

给两个主要原因&#xff0c;这两个原因&#xff0c;如果在递归或者循环里就很容易导致内存超出限制 首先就是误用 如果只是变量的变化实现的话&#xff0c;或者-就可以了&#xff0c;不需要写成AAx的形式&#xff0c;那样会重新开辟一块内存 其次就是值传递 值传递也是会开辟…

在Linux中对Nginx进行安全加固

准备工作 在IP为x.x.x.x的服务器上安装nginx&#xff0c;确保Linux系统为nginx环境。 检查nginx是否配置nginx账号锁定策略 配置nginx账号锁定策略&#xff0c;降低被攻击概率。 第一步&#xff0c;查看nginx的锁定状态。 命令&#xff1a;passwd -S nginx 若结果出现“P…

【gcc】webrtc发送侧计算 丢包率

大神的分析 : 提到: 每当收到cc-feedback或者收到RR-report的时候就能统计出丢包率,在cc-controller中就会调用SendSideBandwidthEstimation::UpdatePacketsLost()去更新丢包率,同时进行码率预估 G:\CDN\rtcCli\m98\src\modules\congestion_controller\goog_cc\send_side_b…

今日arXiv最热NLP大模型论文:像人一样浏览网页执行任务,腾讯AI lab发布多模态端到端Agent

Agent的发展成为了LLM发展的一个热点。只需通过简单指令&#xff0c;Agent帮你完成从输入内容、浏览网页、选择事项、点击、返回等一系列需要执行多步&#xff0c;才能完成的与网页交互的复杂任务。 比如给定任务&#xff1a;“搜索Apple商店&#xff0c;了解iPad智能保护壳Sm…

Qt中的绝对位置与相对位置

在Qt的开发中&#xff0c;有时候需要获取鼠标点击的位置的坐标。在Qt中坐标分为相对坐标和绝对坐标&#xff1b;相对坐标是以控件的左上角为&#xff08;0&#xff0c;0&#xff09;点的坐标系的坐标&#xff0c;绝对坐标是以显示器屏幕的左上角为&#xff08;0&#xff0c;0&a…

蓝桥杯-常用STL(二)

常用STL &#x1f388;1.集合&#x1f388;2.set的基础使用&#x1f52d;2.1引入库&#x1f52d;2.2插入元素&#x1f52d;2.3删除元素&#x1f52d;2.4判断元素是否存在&#x1f52d;2.5遍历元素&#x1f52d;2.6清空 &#x1f388;3.set与结构体 &#x1f388;1.集合 &#x…