美文网首页UIAndroid开发者协程
在 View 上使用挂起函数

在 View 上使用挂起函数

作者: 谷歌开发者 | 来源:发表于2020-10-29 21:48 被阅读0次
    image

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

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

    Android 视图 💘 回调

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

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

    然后还有一些通过接受 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 = true
        titleView.text = "Hi everyone!"
    
        // 等待下一次布局事件的任务,然后才可以获取该视图的高度
        titleView.awaitNextLayout()
    
        // 布局任务被执行
        // 现在,我们可以将视图设置为可见,并其向上平移,然后执行向下的动画
        titleView.isVisible = true
        titleView.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 = true
    
            override 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: ValueAnimator
        imageView.animate().run {
            alpha(0f)
            start()
            awaitEnd()
        }
    
        // #2: RecyclerView smooth scroll
        recyclerView.run {
            smoothScrollToPosition(10)
            // 该方法和其他方法类似,等待当前的滑动完成,我们不需要刻意关注实现
            // 代码可以在文末的引用中找到
            awaitScrollEnd()
        }
    
        // #3: ObjectAnimator
        ObjectAnimator.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 和滑动都完成了,我们开始执行 ObjectAnimator
        ObjectAnimator.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,第三次:450ms
                duration = (repetition + 1) * 150L
                start()
                awaitEnd()
            }
        }
    }
    

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

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

    最后

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

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

    相关文章

      网友评论

        本文标题:在 View 上使用挂起函数

        本文链接:https://www.haomeiwen.com/subject/itwlvktx.html