Kotlin Flow 倒计时

冷流才是正统!!!!!!!!

1. 【有】生命周期感知,倒计时

1.离开页面,倒计时停止,回到页面倒计时继续,例如广告

  1. 可于 Activity / Fragment 中直接使用,没必要封装
    // (属性) 用于取消,确保不会重复收集流
    private var mTickJob: Job? = null

    // (函数),开始倒计时
    private fun countDown() {
        mTickJob?.cancel()
        var duration = 10
        mTickJob = lifecycleScope.launch {
            // 离开目标页面的时候,暂停收集流
            repeatOnLifecycle(Lifecycle.State.STARTED) {
                flow {
                    while (true) {
                        kotlinx.coroutines.delay(1000)
                        emit(null)
                        Log.d("BMO-BMO", "emit...")
                    }
                }.onStart {
                    // 开始,显示倒计时 UI
                    Log.d("BMO-BMO", "开始倒计时 $duration s")
                }.collect {
                    duration -= 1
                    // 更新倒计时 UI
                    Log.d("BMO-BMO", "倒计时 $duration s")
                    if (duration <= 0) {
                        mTickJob?.cancel()
                        // 结束,隐藏倒计时 UI
                        Log.d("BMO-BMO", "倒计时 结束!!")
                    }
                }
            }
        }
    }

1. 【无】生命周期感知,倒计时

可用于发送验证码之类的倒计时

xxx.kt 封装倒计时函数

/**
 * 于协程中实现倒计时
 * @param duration 倒计时时长
 * @param interval 倒计时步长
 * @param scope 协程作用域
 * @param onTick 倒计时变更,回调
 * @param onStart 倒计时开始,回调
 * @param onEnd 倒计时结束,回调
 *
 * @return [Job] 可用于在需要时,取消倒计时
 */
fun countDownCoroutine(
    duration: Int,
    interval: Int = 1,
    scope: CoroutineScope,
    onTick: (Int) -> Unit,
    onStart: ((Int) -> Unit)? = null,
    onEnd: (() -> Unit)? = null,
): Job {
    if (duration <= 0 || interval <= 0) {
        throw IllegalArgumentException("duration or interval can not less than zero")
    }
    return flow {
        for (i in duration downTo 0 step interval) {
            delay((interval * 1000).toLong())
            emit(i)
        }
    }
        .onEach { onTick.invoke(it) }
        .onStart { onStart?.invoke(duration) }
        .onCompletion {
            // 正常结束,才能回调。
            if (it == null) {
                onEnd?.invoke()
            }
        }
        // 确保上游回调,是在主线程回调
        .flowOn(Dispatchers.Main)
        .launchIn(scope)
}

2. 目标页面调用

我们倒计时,建议使用 lifecycleScope, 便不需要 onDestroy 中调用 mTickJob?.cancel()

// (属性) 用于取消,确保不会重复收集流
    private var mCountDownJob: Job? = null

    // (函数),开始倒计时
    private fun countDown() {
        mCountDownJob?.cancel()
        mCountDownJob = countDownCoroutine(duration = 10, scope = lifecycleScope, onTick = {
            Log.d("BMO-BMO", "更新 UI ,剩余时间 $it s")
        }, onStart = {
            Log.d("BMO-BMO", "倒计时开始 ,剩余时间 $it s")
        }, onEnd = {
            Log.d("BMO-BMO", "倒计时结束 !!")
            mCountDownJob?.cancel()
        })
    }

关于协程和 Kotlin Flow 的知识点,诸位自行查阅,嗯。

3. 【有】生命周期感知,TickFlow 秒触发器

如果页面需要显示手机本地时间,或者例如每 1 s 就干点什么事情,可以试试
repeatOnLifecycle 使得离开页面,不会收集流,job 直接被取消掉

函数

/**
 *
 * @param lifecycleOwner 直接传 Activity / Fragment
 * @param onTick 每 [interval] 毫秒,滴答一下
 */
fun tickFlow(
    interval: Long = 1000,
    lifecycleOwner: LifecycleOwner,
    onTick: () -> Unit,
): Job {
    return lifecycleOwner.lifecycleScope.launch {
        lifecycleOwner.repeatOnLifecycle(Lifecycle.State.STARTED) {
            flow {
                while (true) {
                    delay(interval)
                    emit(null)
                }
            }.collect {
                onTick.invoke()
            }
        }
    }
}

Activity / Fragment 中调用

// (属性) 用于取消,确保不会重复收集流
    private var mTickJob: Job? = null

    // 懒加载,复用对象
    private val mTickDate by lazy { Date() }

    // (函数),开始滴答
    private fun startTick() {
        mTickJob?.cancel()
        mTickJob = tickFlow(lifecycleOwner = this) {
            mTickDate.time = System.currentTimeMillis()
            Log.d(
                "BMO-BMO",
                "当前时间: ${TimeDateHelper.dateToString(mTickDate, TimeDateHelper.longerPattern)}"
            )
        }
    }

    /**
     * Date 转 字符串
     */
    fun dateToString(date:Date, pattern: String = shortPattern):String {
        val simpleDateFormat = SimpleDateFormat(pattern)
        return simpleDateFormat.format(date)
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,319评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,801评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,567评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,156评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,019评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,090评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,500评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,192评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,474评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,566评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,338评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,212评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,572评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,890评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,169评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,478评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,661评论 2 335

推荐阅读更多精彩内容