Kotlin 协程版的 AutoDispose

Kotlin 协程版的 AutoDispose_第1张图片


大家一定用过 RxJava,也一定知道用 RxJava 发了个任务,任务还没结束页面就被关闭了,如果任务迟迟不回来,页面就会被泄露;如果任务后面回来了,执行回调更新 UI 的时候也会大概率空指针。

因此大家一定会用到 Uber 的开源框架 AutoDispose。

什么?你说你没用?好吧,那就没用吧。。我是不会介绍它的。⊙﹏⊙|||。怎么可能。(~ ̄▽ ̄)~。。它其实就是利用 ViewOnAttachStateChangeListener ,当 View 被拿下的时候,我们就取消所有之前用 RxJava 发出去的请求。

 
   
  1.  static final class Listener extends MainThreadDisposable implements View.OnAttachStateChangeListener {

  2.    private final View view;

  3.    private final CompletableObserver observer;


  4.    Listener(View view, CompletableObserver observer) {

  5.      this.view = view;

  6.      this.observer = observer;

  7.    }


  8.    @Override public void onViewAttachedToWindow(View v) { }


  9.    @Override public void onViewDetachedFromWindow(View v) {

  10.      if (!isDisposed()) {

  11.      //看到没看到没看到没?

  12.        observer.onComplete();

  13.      }

  14.    }


  15.    @Override protected void onDispose() {

  16.      view.removeOnAttachStateChangeListener(this);

  17.    }

  18.  }

好了,我最近在想我们用协程其实也会有这样的问题呀:

 
   
  1. button.onClick {

  2.    try {

  3.        val req = Request()

  4.        val resp = async { sendRequest(req) }.await()

  5.        updateUI(resp)

  6.    } catch (e: Exception) {

  7.        e.printStackTrace()

  8.    }

  9. }

如果 await 返回结果之前我们就退出了当前的 Activity 那么,后面 updateUI 就要凉凉。这就尴尬了。不过问题不大,照猫画虎谁不会,我们也可以搞一个 onClickAutoDisposable 嘛。

 
   
  1. fun View.onClickAutoDisposable (

  2.        context: CoroutineContext = Dispatchers.Main,

  3.        handler: suspend CoroutineScope.(v: android.view.View?) -> Unit

  4. ) {

  5.    setOnClickListener { v ->

  6.        GlobalScope.launch(context, CoroutineStart.DEFAULT) {

  7.            handler(v)

  8.        }.asAutoDisposable(v)

  9.    }

  10. }

第一步,不要脸的先抄 Anko 的 onClick,不同之处在于我们改了个名 XD。啊,还有我们加了个 .asAutoDisposable(v),大家就假装有这个方法吧。。。

(╬ ̄皿 ̄)=○ 假装个头啊,假装就完成功能的话还要程序员干什么。。让产品假装一下不就行了。。

OK OK,咱们下面来实现它。。想想, GlobalScope.launch 其实返回的是一个 Job,所以嘛,我们给 Job 搞一个扩展方法不就得了。

 
   
  1. fun Job.asAutoDisposable(view: View) = AutoDisposableJob(view, this)

第二步,我们再偷偷的创建一个类,叫 AutoDisposableJob,抄一下前面的 Listener

 
   
  1. class AutoDisposableJob(private val view: View, private val wrapped: Job)

  2.    //我们实现了 Job 这个接口,但没有直接实现它的方法,而是用 wrapped 这个成员去代理这个接口

  3.     : Job by wrapped, OnAttachStateChangeListener {

  4.    override fun onViewAttachedToWindow(v: View?) = Unit


  5.    override fun onViewDetachedFromWindow(v: View?) {

  6.        //当 View 被移除的时候,取消协程

  7.        cancel()

  8.        view.removeOnAttachStateChangeListener(this)

  9.    }


  10.    private fun isViewAttached() =

  11.            Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT && view.isAttachedToWindow || view.windowToken != null


  12.    init {

  13.        if(isViewAttached()) {

  14.            view.addOnAttachStateChangeListener(this)

  15.        } else {

  16.            cancel()

  17.        }


  18.        //协程执行完毕时要及时移除 listener 免得造成泄露

  19.        invokeOnCompletion() {

  20.            view.removeOnAttachStateChangeListener(this)

  21.        }

  22.    }

  23. }

这样的话,我们就可以使用这个扩展了:

 
   
  1. button.onClickAutoDisposable{

  2.    try {

  3.        val req = Request()

  4.        val resp = async { sendRequest(req) }.await()

  5.        updateUI(resp)

  6.    } catch (e: Exception) {

  7.        e.printStackTrace()

  8.    }

  9. }

button 这个对象从 window 上撤下来的时候,我们的协程就会收到 cancel 的指令。

好的,我又凭实力忽悠了本周的文章。。。


转载请注明出处:微信公众号 Kotlin

Kotlin 协程版的 AutoDispose_第2张图片

你可能感兴趣的:(Kotlin 协程版的 AutoDispose)