Kotlin 协程 让我们可以用同步代码来建立异步问题的模型。这是非常好的特性,但是目前大部分用例都专注于 I/O 任务或是并发操作。其实协程不仅在处理跨线程的问题有优势,还可以用来处理同一线程中的异步问题。

我认为有一个地方可以真正从中受益,那就是在 Android 视图系统中使用协程。

Android 视图 回调

Android 视图系统中尤其热衷于使用回调: 目前在 Android Framework 中,view 和 widgets 类中的回调有 80+ 个,在 Jetpack 中回调的数目更是超过了 200 个 (这里也包含了没有界面的依赖库)。

最常见的用法有以下几项:

  • AnimatorListener 获取动画结束相关的事件
  • RecyclerView.OnScrollListener 获取滑动状态变更事件
  • View.OnLayoutChangeListener 获取 View 布局改变的事件

然后还有一些通过接受 Runnable 来执行异步操作的API,比如 View.post()、View.postDelayed() 等等。

正是因为 Android 上的 UI 编程从根本上就是异步的,所以造成了如此之多的回调。从测量、布局、绘制,到调度插入,整个过程都是异步的。通常情况下,一个类 (通常是 View) 调用系统方法,一段时间之后系统来调度执行,然后通过回调触发监听。

KTX 扩展方法

上述提及的 API,在 Jetpack 中都增加了扩展方法来提高开发效率。其中 View.doOnPreDraw()方法是我最喜欢的一个,该方法对等待下一次绘制被执行进行了极大的精简。其实还有很多我常用的方法,比如 View.doOnLayout()Animator.doOnEnd()

但是这些扩展方法也是仅止步于此,他们只是将旧风格的回调 API 改成了 Kotlin 中比较友好的基于 lambda 风格的 API。虽然用起来很优雅,但我们只是在用另一种方式处理回调,这还是没有解决复杂的 UI 的回调嵌套问题。既然我们在讨论异步操作,那在这种情况下,我们可以使用协程优化这些问题么?

使用协程解决问题

这里假定您已经对协程有一定的理解,如果接下来的内容对您来说会有些陌生,可以通过我们今年早期的系列文章进行回顾: 在 Android 开发中使用协程 | 背景介绍

挂起函数 (Suspending functions) 是协程的基础组成部分,它允许我们以非阻塞的方式编写代码。这种特性非常适用于我们处理 Android UI,因为我们不想阻塞主线程,阻塞主线程会带来性能上的问题,比如: jank

suspendCancellableCoroutine

在 Kotlin 协程库中,有很多协程的构造器方法,这些构造器方法内部可以使用挂起函数来封装回调的 API。最主要的 API 是 suspendCoroutine()suspendCancellableCoroutine(),后者是可以被取消的。

我们推荐始终使用 suspendCancellableCoroutine(),因为这个方法可以从两个维度处理协程的取消操作:

#1: 可以在异步操作完成之前取消协程。如果某个 view 从它所在的层级中被移除,那么根据协程所处的作用域 (scope),它有可能会被取消。举个例子: Fragment 返回出栈,通过处理取消事件,我们可以取消异步操作,并清除相关引用的资源。

#2: 在协程被挂起的时候,异步 UI 操作被取消或者抛出异常。并不是所有的操作都有已取消或出错的状态,但是这些操作有。就像后面 Animator 的示例中那样,我们必须把这些状态传递到协程中,让调用者可以处理错误的状态。

等待 View 被布局完成

让我们看一个例子,它封装了一个等待 View 传递下一次布局事件的任务 (比如说,我们改变了一个 TextView 中的内容,需要等待布局事件完成后才能获取该控件的新尺寸):

suspend fun View.awaitNextLayout() = suspendCancellableCoroutine<Unit> { cont ->// 这里的 lambda 表达式会被立即调用,允许我们创建一个监听器val listener = object : View.OnLayoutChangeListener {override fun onLayoutChange(...) {// 视图的下一次布局任务被调用// 先移除监听,防止协程泄漏view.removeOnLayoutChangeListener(this)// 最终,唤醒协程,恢复执行cont.resume(Unit)}}// 如果协程被取消,移除该监听cont.invokeOnCancellation { removeOnLayoutChangeListener(listener) }// 最终,将监听添加到 view 上addOnLayoutChangeListener(listener)// 这样协程就被挂起了,除非监听器中的 cont.resume() 方法被调用}

此方法仅支持协程中一个维度的取消 (#1 操作),因为布局操作没有错误状态供我们监听。

接下来我们就可以这样使用了:

viewLifecycleOwner.lifecycleScope.launch {// 将该视图设置为不可见,再设置一些文字titleView.isInvisible = truetitleView.text = "Hi everyone!"// 等待下一次布局事件的任务,然后才可以获取该视图的高度titleView.awaitNextLayout()// 布局任务被执行// 现在,我们可以将视图设置为可见,并其向上平移,然后执行向下的动画titleView.isVisible = truetitleView.translationY = -titleView.height.toFloat()titleView.animate().translationY(0f)
}

我们为 View 的布局创建了一个 await 函数。用同样的方法可以替代很多常见的回调,比如 doOnPreDraw(),它是在 View 得到绘制时调用的方法;再比如 postOnAnimation(),在动画的下一帧开始时调用的方法,等等。

作用域

不知道您有没有发现这样一个问题,在上面的例子中,我们使用了 lifecycleScope 来启动协程,为什么要这样做呢?

为了避免发生内存泄漏,在我们操作 UI 的时候,选择合适的作用域来运行协程是极其重要的。幸运的是,我们的 View 有一些范围合适的 Lifecycle。我们可以使用扩展属性 lifecycleScope 来获得一个绑定生命周期的 CoroutineScope

LifecycleScope 被包含在 AndroidX 的 lifecycle-runtime-ktx 依赖库中,可以在这里找到 更多信息。

我们最常用的生命周期的持有者 (lifecycle owner) 就是 Fragment 中的 viewLifecycleOwner,只要加载了 Fragment 的视图,它就会处于活跃状态。一旦 Fragment 的视图被移除,与之关联的 lifecycleScope 就会自动被取消。又由于我们已经为挂起函数中添加了对取消操作的支持,所以 lifecycleScope 被取消时,所有与之关联的协程都会被清除。

等待 Animator 执行完成

我们再来看一个例子来加深理解,这次是等待 Animator 执行结束:

suspend fun Animator.awaitEnd() = suspendCancellableCoroutine<Unit> { cont ->// 增加一个处理协程取消的监听器,如果协程被取消,// 同时执行动画监听器的 onAnimationCancel() 方法,取消动画cont.invokeOnCancellation { cancel() }addListener(object : AnimatorListenerAdapter() {private var endedSuccessfully = trueoverride fun onAnimationCancel(animation: Animator) {// 动画已经被取消,修改是否成功结束的标志endedSuccessfully = false}override fun onAnimationEnd(animation: Animator) {// 为了在协程恢复后的不发生泄漏,需要确保移除监听animation.removeListener(this)if (cont.isActive) {// 如果协程仍处于活跃状态if (endedSuccessfully) {// 并且动画正常结束,恢复协程cont.resume(Unit)} else {// 否则动画被取消,同时取消协程cont.cancel()}}}})
}

这个方法支持两个维度的取消,我们可以分别取消动画或者协程:

#1: 在 Animator 运行的时候,协程被取消 。我们可以通过 invokeOnCancellation 回调方法来监听协程何时被取消,这能让我们同时取消动画。

#2: 在协程被挂起的时候,Animator 被取消 。我们通过 onAnimationCancel() 回调来监听动画被取消的事件,通过调用协程的 cancel() 方法来取消挂起的协程。

这就是使用挂起函数等待方法执行来封装回调的基本使用了。

组合使用

到这里,您可能有这样的疑问,"看起来不错,但是我能从中收获什么呢?" 单独使用其中某个方法,并不会产生多大的作用,但是如果把它们组合起来,便能发挥巨大的威力。

下面是一个使用 Animator.awaitEnd() 来依次运行 3 个动画的示例:

viewLifecycleOwner.lifecycleScope.launch {ObjectAnimator.ofFloat(imageView, View.ALPHA, 0f, 1f).run {start()awaitEnd()}ObjectAnimator.ofFloat(imageView, View.TRANSLATION_Y, 0f, 100f).run {start()awaitEnd()}ObjectAnimator.ofFloat(imageView, View.TRANSLATION_X, -100f, 0f).run {start()awaitEnd()}
}

这是一个很常见的使用案例,您可以把这些动画放进 AnimatorSet 中来实现同样的效果。

但是这里使用的方法适用于不同类型的异步操作: 我们使用一个 ValueAnimator,一个 RecyclerView 的平滑滚动,以及一个 Animator 来举例:

viewLifecycleOwner.lifecycleScope.launch {// #1: ValueAnimatorimageView.animate().run {alpha(0f)start()awaitEnd()}// #2: RecyclerView smooth scrollrecyclerView.run {smoothScrollToPosition(10)// 该方法和其他方法类似,等待当前的滑动完成,我们不需要刻意关注实现// 代码可以在文末的引用中找到awaitScrollEnd()}// #3: ObjectAnimatorObjectAnimator.ofFloat(textView, View.TRANSLATION_X, -100f, 0f).run {start()awaitEnd()}
}

试着用 AnimatorSet 实现一下吧 !如果不用协程,那就意味着我们要监听每一个操作,在回调中执行下一个操作,这回调层级想想都可怕。

通过把不同的异步操作转换为协程的挂起函数,我们获得了简洁明了地编排它们的能力。

我们还可以更进一步...

**如果我们希望 ValueAnimator 和平滑滚动同时开始,然后在两者都完成之后启动 ObjectAnimator,该怎么做呢?**那么在使用了协程之后,我们可以使用 async() 来并发地执行我们的代码:

viewLifecycleOwner.lifecycleScope.launch {val anim1 = async {imageView.animate().run {alpha(0f)start()awaitEnd()}}val scroll = async {recyclerView.run {smoothScrollToPosition(10)awaitScrollEnd()}}// 等待以上两个操作全部完成anim1.await()scroll.await()// 此时,anim1 和滑动都完成了,我们开始执行 ObjectAnimatorObjectAnimator.ofFloat(textView, View.TRANSLATION_X, -100f, 0f).run {start()awaitEnd()}
}

但是如果您还想让滚动延迟执行怎么办呢? (类似 Animator.startDelay 方法) 那么使用协程也有很好的实现,我们可以用 delay() 方法:

viewLifecycleOwner.lifecycleScope.launch {val anim1 = async {// ...}val scroll = async {// 我们希望在 anim1 完成后,延迟 200ms 执行滚动delay(200)recyclerView.run {smoothScrollToPosition(10)awaitScrollEnd()}}// …
}

如果我们想重复动画,那么我们可以使用 repeat() 方法,或者使用 for 循环实现。下面是一个 view 淡入淡出 3 次的例子:

viewLifecycleOwner.lifecycleScope.launch {repeat(3) {ObjectAnimator.ofFloat(textView, View.ALPHA, 0f, 1f, 0f).run {start()awaitEnd()}}
}

您甚至可以通过重复计数来实现更精妙的功能。假设您希望淡入淡出在每次重复中逐渐变慢:

viewLifecycleOwner.lifecycleScope.launch {repeat(3) { repetition ->ObjectAnimator.ofFloat(textView, View.ALPHA, 0f, 1f, 0f).run {// 第一次执行持续 150ms,第二次:300ms,第三次:450msduration = (repetition + 1) * 150Lstart()awaitEnd()}}
}

在我看来,这就是在 Android 视图系统中使用协程能真正发挥作用的地方。我们就算不去组合不同类型的回调,也能创建复杂的异步变换,或是将不同类型的动画组合起来。

通过使用与我们应用中数据层相同的协程开发原语,还能使 UI 编程更便捷。对于刚接触代码的人来说, await 方法要比看似会断开的回调更具可读性。

最后

希望通过本文,您可以进一步思考协程还可以在哪些其他的 API 中发挥作用。

接下来的文章中,我们将探讨如何使用协程来组织一个复杂的变换动画,其中也包括了一些常见 View 的实现,感兴趣的读者请继续关注我们的更新。

中调用view_在 View 上使用挂起函数相关推荐

  1. 在网页中调用摄像头实现拍照上传 - 高拍仪二次开发

    在网页中调用摄像头实现拍照上传 高拍仪二次开发     在一些公共部门的办事处,比如银行.护照办理中心.税务等,我们可能会注意到办公桌上摆着这样一台机器.办公人员用它拍摄各种证件.文件.表格,有时候还 ...

  2. c++ 二次开发 良田高拍仪_在网页中调用摄像头实现拍照上传 - 高拍仪二次开发...

    来源于  https://blog.csdn.net/weixin_40659738/article/details/78252562 在网页中调用摄像头实现拍照上传 高拍仪二次开发 在一些公共部门的 ...

  3. weUI中调用手机摄像头拍照上传

    //图片上传 <div class="weui-cells weui-cells_form" style="margin-top: 0px;">&l ...

  4. 利用JNI技术在Android中调用C++形式的OpenGL ES 2.0函数

    1.                 打开Eclipse,File-->New-->Project--->Android-->AndroidApplication Projec ...

  5. java里上下文对象,java-在百里香模板中的Web上下文对象上发出...

    我试图在thymeleaf 3.0.3和Spring Boot 1.5.1的模板中调用Web上下文对象上的方法,例如#request和#response. 我不断收到这样的错误: org.spring ...

  6. 【Java】-在Java中调用大漠插件

    目录 在Java中调用大漠插件步骤 常见问题 Java与Dll函数的数据通信(一个比较大的坑) 注册了大漠高版本后,如何更换为低版本? Description: 80020010 / 无效的被呼叫方. ...

  7. vb调用本地html,在VB中调用HTMLHELP文件VB -电脑资料

    在 VB 中调用HTMLHELP文件 湖北省襄樊市劳动保险处 闫东 ---- HTML帮助文件是 WINDOWS 基本帮助系统的一种新标准,在WINDOWS 98中我们到处都可以看到它的身影, 在VB ...

  8. php 类中调用另类,PHP return语句另类用法不止是在函数中,return语句_PHP教程

    PHP return语句另类用法不止是在函数中,return语句 分享下PHP return语句的另一个作用,在bbPress的代码中看到的一个奇葩使用方法. 一直以为,return只能出现在函数中, ...

  9. python中调用C++函数

    python中调用C++函数 无参调用 单变量传入与返回调用 numpy数组传入与返回调用 c++类调用 用python写不香吗?还这么复杂调用C++? 一. 无参调用 在python中调用无参数和无 ...

  10. Effective C++条款09:绝不在构造和析构过程中调用virtual函数

    Effective C++条款09:绝不在构造和析构过程中调用virtual函数(Never call virtual functions during construction or destruc ...

最新文章

  1. MySQL建表枚举分区SQL,【mysql备份】02、Xtrabackup备份mysql
  2. Spring Boot Security
  3. CoreData / MagicalRecord
  4. spring配置详解-模块化配置
  5. 50 jQuery绑定事件 阻止默认事件发生 内置动画 each data
  6. 【Pytorch神经网络理论篇】 21 信息熵与互信息:联合熵+条件熵+交叉熵+相对熵/KL散度/信息散度+JS散度
  7. [Pyramid 杂记]Static Routes,静态路由是用来干什么的?
  8. 基于JavaFX实现的数据库学生管理系统
  9. linux上安装osg_ubuntu 环境 安装OSG
  10. 文件操作03 - 零基础入门学习C语言62
  11. Java学习----方法的覆盖
  12. JSONObject 与 JSONArray 使用
  13. 锅炉的计算机控制系统设计,余热锅炉计算机控制系统设计与开发
  14. 入坑codewars第五天-Dubstep、Regex validate PIN code
  15. Linux中将两块新硬盘合并成一个,挂载到/data目录下
  16. Android——给button添加图片
  17. JavaScript|日期格式化、今天、昨天、明天和某天
  18. 我做的游戏终于上线了----三国Q传!!!
  19. 抱团股会一直涨?无脑执行大小盘轮动策略,轻松跑赢指数5倍【附Python代码】
  20. opencv 稀疏光流 稠密光流

热门文章

  1. 为Bootstrap模态对话框添加拖拽移动功能
  2. Windows 2012 R2 操作系统搭建DHCP-HA集群
  3. 负载均衡之LVS详解
  4. 新年跨出第一步:人工智能实施这样做!
  5. Windows10下安装原生TensorFlow GPU版
  6. larave 5 could not be opened: failed to open stream: Permission denied
  7. 集合框架(数据结构之栈和队列)
  8. tcpdump 的TCP输出结果详解
  9. VS2012发布网站IIS配置
  10. js的this作用域