当前位置:首页 > 谈天说地

一文了解Android ViewModelScope 如何自动取消协程

34资源网2022-07-04446

先看一下 viewmodel 中的 viewmodelscope 是何方神圣

val viewmodel.viewmodelscope: coroutinescope
        get() {
            val scope: coroutinescope? = this.gettag(job_key)
            if (scope != null) {
                return scope
            }
            return settagifabsent(job_key,
                closeablecoroutinescope(supervisorjob() + dispatchers.main.immediate))
        }

可以看到这个是一个扩展方法,

再点击 settagifabsent 方法进去

 <t> t settagifabsent(string key, t newvalue) {
        t previous;
        synchronized (mbagoftags) {
            previous = (t) mbagoftags.get(key);//第一次肯定为null
            if (previous == null) {
                mbagoftags.put(key, newvalue);//null 存储
            }
        }
        t result = previous == null ? newvalue : previous;
        if (mcleared) {//判断是否已经clear了
            // it is possible that we'll call close() multiple times on the same object, but
            // closeable interface requires close method to be idempotent:
            // "if the stream is already closed then invoking this method has no effect." (c)
            closewithruntimeexception(result);
        }
        return result;
    }

可以看到 这边 会把 我们的 viewmodel 存储到 viewmodel 内的 mbagoftags 中

这个 mbagoftags 是

    private final map<string, object> mbagoftags = new hashmap<>();

这个时候 我们 viewmodel 就会持有 我们 viewmodelscope 的协程 作用域了。那..这也只是 表述了 我们 viewmodelscope 存在哪里而已,什么时候清除呢?

先看一下 viewmodel 的生命周期:

可以看到 viewmodel 的生命周期 会在 activity ondestory 之后会被调用。那...具体哪里调的?

翻看源码可以追溯到 componentactivity 的默认构造器内

 public componentactivity() {
     /*省略一些*/
        getlifecycle().addobserver(new lifecycleeventobserver() {
            @override
            public void onstatechanged(@nonnull lifecycleowner source,
                    @nonnull lifecycle.event event) {
                if (event == lifecycle.event.on_destroy) {
                    if (!ischangingconfigurations()) {
                        getviewmodelstore().clear();
                    }
                }
            }
        });
  }

可以看到内部会通对 lifecycle 添加一个观察者,观察当前 activity 的生命周期变更事件,如果走到了 destory ,并且 本次 destory 并非由于配置变更引起的,才会真正调用 viewmodelstore 的 clear 方法。

跟进 clear 方法看看:

public class viewmodelstore {
    private final hashmap<string, viewmodel> mmap = new hashmap<>();

    /**
     *  clears internal storage and notifies viewmodels that they are no longer used.
     */
    public final void clear() {
        for (viewmodel vm : mmap.values()) {
            vm.clear();
        }
        mmap.clear();
    }
}

可以看到这个 viewmodelstore 内部实现 用 hashmap 存储 viewmodel

于是在 clear 的时候,会逐个遍历调用 clear方法,再次跟进 viewmodel 的 clear 方法

 @mainthread
    final void clear() {
        mcleared = true;
        // since clear() is final, this method is still called on mock objects
        // and in those cases, mbagoftags is null. it'll always be empty though
        // because settagifabsent and gettag are not final so we can skip
        // clearing it
        if (mbagoftags != null) {
            synchronized (mbagoftags) {
                for (object value : mbagoftags.values()) {
                    // see comment for the similar call in settagifabsent
                    closewithruntimeexception(value);
                }
            }
        }
        oncleared();
    }

可以发现我们最初 存放 viewmodelscope 的 mbagoftags

这里面的逻辑 就是对 mbagoftags 存储的数据 挨个提取出来并且调用 closewithruntimeexception

跟进 closewithruntimeexception:

 private static void closewithruntimeexception(object obj) {
        if (obj instanceof closeable) {
            try {
                ((closeable) obj).close();
            } catch (ioexception e) {
                throw new runtimeexception(e);
            }
        }
    }

该方法内会逐个判断 对象是否实现 closeable 如果实现就会调用这个接口的 close 方法,

再回到最初 我们 viewmodel 的扩展方法那边,看看我们 viewmodelscope 的真正面目

internal class closeablecoroutinescope(context: coroutinecontext) 
    : closeable, coroutinescope {
    override val coroutinecontext: coroutinecontext = context

    override fun close() {
        coroutinecontext.cancel()
    }
}

可以明确的看到 我们的 viewmodelscope 实现了 closeable 并且充写了 close 方法,

close 方法内的实现 会对 协程上下文进行 cancel。

至此我们 可以大致整理一下:

  • viewmodelscope 是 viewmodel 的扩展成员,该对象是 closeablecoroutinescope,并且实现了 closeable 接口
  • viewmodelscope 存储在 viewmodel 的 名叫 mbagoftags 的hashmap中 啊
  • viewmodel 存储在 activity 的 viewmodelstore 中,并且会监听 activity 的 lifecycle 的状态变更,在on_destroy 且 非配置变更引起的事件中 对 viewmodelstore 进行清空
  • viewmodelstore 清空会对 viewmodelstore 内的所有 viewmodel 逐个调用 clear 方法。
  • viewmodel的clear方法会对 viewmodel的 mbagoftags 内存储的对象进行调用 close 方法(该对象需实现closeable 接口)
  • 最终会会调用 我们 viewmodelscope 的实现类 closeablecoroutinescope 的 close 方法中。close 方法会对协程进行 cancel。

到此这篇关于一文了解android viewmodelscope 如何自动取消协程的文章就介绍到这了,更多相关android viewmodel scope 取消协程内容请搜索萬仟网以前的文章或继续浏览下面的相关文章希望大家以后多多支持萬仟网!

看完文章,还可以扫描下面的二维码下载快手极速版领4元红包

快手极速版二维码

快手极速版新人见面礼

除了扫码领红包之外,大家还可以在快手极速版做签到,看视频,做任务,参与抽奖,邀请好友赚钱)。

邀请两个好友奖最高196元,如下图所示:

快手极速版邀请好友奖励

扫描二维码推送至手机访问。

版权声明:本文由34楼发布,如需转载请注明出处。

本文链接:https://www.34l.com/post/18454.html

分享给朋友:

相关文章

50句非常励志的短句,正能量的励志句子

50句非常励志的短句,正能量的励志句子

1、人非要经历一番不同平时的劫难才能脱胎换骨,成为真正能解决问题的人。…

现在小县城做什么创业比较好?

现在小县城做什么创业比较好?

目前在小县城做什么赚钱创业比较好呢?我相信大家都在想这个问题吧?其实小县城是一个不错的经济市场,主要看你做什么项目创业了。下面小编收集了几个项目,希望大家喜欢。…

曹德旺的经典语录分享

曹德旺的经典语录分享

曹德旺,福耀玻璃集团创始人、董事长。中国反倾销胜诉第一人、中国首善、安永企业家奖首位华人获得者 。福耀玻璃目前是中国第一、世界第二大汽车玻璃供应商。他是不行贿的企业家,自称“没送过一盒月饼”,以人格做事;他是行善的佛教徒,从1983年第一次…

联想新手机什么时候上市(联想2021即将上市新款笔记本)

联想新手机什么时候上市(联想2021即将上市新款笔记本)

11月8日,联想中国区手机业务部总经理发布了一则新机预告:摩托罗拉edge X的发布已进入倒计时阶段,在骁龙898处理器即将发布之际预告新机,很大可能预示着首发权已到手。…

个人微信号怎么更改(设置里的微信密码)

个人微信号怎么更改(设置里的微信密码)

有多少人和我一样,起了一个非常非常中二的名字。当初的年少无知终于可以后悔了。微信号终于!终于!终于可以修改微信号了。(IOS用户请先等等,现在仅支持安卓用户哦) 微信最近推出新功能,允许用户每年修改一次微信号。 TOP1:首先把微信更新到…