在 Android 上测试 Kotlin 数据流

文章目录

    • 一 创建虚构数据提供方
    • 二 在测试中断言数据流发出
      • 测试期间持续收集
    • 三 测试 StateFlow
      • 使用 stateIn 创建的 StateFlow

转自:
https://developer.android.google.cn/kotlin/flow/test?hl=zh-cn#producer

与数据流进行通信的单元或模块的测试方式取决于受测对象使用数据流作为输入还是输出

  • 如果受测对象观察到数据流,您可以在虚构依赖项中生成数据流,而这些可以通过测试进行控制
  • 如果单元或模块公开了数据流,您可以读取并验证测试中的数据流所发出的一个或多个数据项

一 创建虚构数据提供方

当受测对象是数据流使用方时,一种常见的测试方法是用虚构实现替换提供方。

class MyFakeRepository : MyRepository {fun observeCount() = flow {emit(ITEM_1)}
}
@Test
fun myTest() {// Given a class with fake dependencies:val sut = MyUnitUnderTest(MyFakeRepository())// Trigger and verify...
}

二 在测试中断言数据流发出

1、某些测试,您只需要检查来自数据流的第一个发出项或有限数量的项

@Test
fun myRepositoryTest() = runTest {// Given a repository that combines values from two data sources:val repository = MyRepository(fakeSource1, fakeSource2)// When the repository emits a valueval firstItem = repository.counter.first() // Returns the first item in the flow// Then check it's the expected itemassertEquals(ITEM_1, firstItem)
}

2、如果该测试需要检查多个值,则调用 toList() 会使数据流等待数据源发出其所有值,然后以列表形式返回这些值

@Test
fun myRepositoryTest() = runTest {// Given a repository with a fake data source that emits ALL_MESSAGESval messages = repository.observeChatMessages().toList()// When all messages are emitted then they should be ALL_MESSAGESassertEquals(ALL_MESSAGES, messages)
}

3、对于需要更复杂地收集数据项或未返回有限数据项的数据流,您可使用 Flow API 选取并转换数据项。

// Take the second item
outputFlow.drop(1).first()// Take the first 5 items
outputFlow.take(5).toList()// Takes the first item verifying that the flow is closed after that
outputFlow.single()// Finite data streams
// Verify that the flow emits exactly N elements (optional predicate)
outputFlow.count()
outputFlow.count(predicate)

测试期间持续收集

在测试中使用虚构实现时,您可以创建一个收集协程,该协程会持续接收 Repository 中的值

class Repository(private val dataSource: DataSource) {fun scores(): Flow<Int> {return dataSource.counts().map { it * 10 }}
}class FakeDataSource : DataSource {private val flow = MutableSharedFlow<Int>()suspend fun emit(value: Int) = flow.emit(value)override fun counts(): Flow<Int> = flow
}

可以创建一个收集协程,该协程会持续接收 Repository 中的值。

@Test
fun continuouslyCollect() = runTest {val dataSource = FakeDataSource()val repository = Repository(dataSource)val values = mutableListOf<Int>()backgroundScope.launch(UnconfinedTestDispatcher(testScheduler)) {repository.scores().toList(values)}dataSource.emit(1)assertEquals(10, values[0]) // Assert on the list contentsdataSource.emit(2)dataSource.emit(3)assertEquals(30, values[2])assertEquals(3, values.size) // Assert the number of items collected
}

由于此处 Repository 公开的数据流永远无法完成,因此收集它的 toList 调用永远不会返回。
使用 Turbine 库
第三方 Turbine 库提供了一个用于创建收集协程的便捷 API,以及用于测试数据流的其他便捷功能

@Test
fun usingTurbine() = runTest {val dataSource = FakeDataSource()val repository = Repository(dataSource)repository.scores().test {// Make calls that will trigger value changes only within test{}dataSource.emit(1)assertEquals(10, awaitItem())dataSource.emit(2)awaitItem() // Ignore items if needed, can also use skip(n)dataSource.emit(3)assertEquals(30, awaitItem())}
}

三 测试 StateFlow

StateFlow 是一种可观察的数据存储器,可以收集这种存储器来以数据流的形式观察它随时间变化的存储值

以下 ViewModel 会从 Repository 收集值,并在 StateFlow 中将值提供给界面

class MyViewModel(private val myRepository: MyRepository) : ViewModel() {private val _score = MutableStateFlow(0)val score: StateFlow<Int> = _score.asStateFlow()fun initialize() {viewModelScope.launch {myRepository.scores().collect { score ->_score.value = score}}}
}

此 Repository 的虚构实现可能如下:

class FakeRepository : MyRepository {private val flow = MutableSharedFlow<Int>()suspend fun emit(value: Int) = flow.emit(value)override fun scores(): Flow<Int> = flow
}

使用此虚构实现测试 ViewModel 时,您可以从虚构实现发出值,以在 ViewModel 的 StateFlow 中触发更新,然后对更新后的 value 断言:

@Test
fun testHotFakeRepository() = runTest {val fakeRepository = FakeRepository()val viewModel = MyViewModel(fakeRepository)assertEquals(0, viewModel.score.value) // Assert on the initial value// Start collecting values from the RepositoryviewModel.initialize()// Then we can send in values one by one, which the ViewModel will collectfakeRepository.emit(1)assertEquals(1, viewModel.score.value)fakeRepository.emit(2)fakeRepository.emit(3)assertEquals(3, viewModel.score.value) // Assert on the latest value
}

使用 stateIn 创建的 StateFlow

ViewModel 使用 MutableStateFlow 存储 Repository 中的数据流发出的最新值。这是一种常见的模式,通常通过使用 stateIn 运算符以更简单的方式实现,该运算符会将冷数据流转换为热 StateFlow:

class MyViewModelWithStateIn(myRepository: MyRepository) : ViewModel() {val score: StateFlow<Int> = myRepository.scores().stateIn(viewModelScope, SharingStarted.WhileSubscribed(5000L), 0)
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/120943.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

算法通关村第十二关白银挑战——仅仅反转英文字母问题解析

大家好&#xff0c;我是怒码少年小码。 今天做道经典的字符串算法题目。 仅仅反转字母 LeetCode 917&#xff1a;给你一个字符串 s &#xff0c;根据下述规则反转字符串&#xff1a; 所有非英文字母保留在原有位置。所有英文字母&#xff08;小写或大写&#xff09;位置反转…

Tomcat运维以及优化

Tomcat常用运维命令 # 查看版本/opt/data/app/tomcat-9.0.82/bin/catalina.sh version## 启动 /opt/data/app/tomcat-9.0.82/bin/startup.sh # 停止 /opt/data/app/tomcat-9.0.82/bin/shutdown.sh调整JVM 参数 方式1 vim /opt/data/app/tomcat-9.0.82/bin/catalina.sh # OS…

硬件加速绘制基础知识

目录 前置知识&#xff1a; OPENGL 和OPENGL ELS的关系 EGLSurface和EGLContext关系 上下文保存着状态集&#xff0c;什么是状态集&#xff1f; Thread和EGLSurface&#xff0c;EGLContext关系 具体api Display和EGL关系 Surface和EGLSurface的关系 大致初始化流程 绘…

以技术创新“谋局”,洗碗机市场的风向变了

在知乎、小红书等知识社区上&#xff0c;时常可以看到一些有趣的“种草”帖&#xff1a;扫地机器人和洗碗机&#xff0c;被并称为“解放人类双手的家电产品”。 特别是洗碗机的话题下&#xff0c;不少对清洁、节能、烘干、容量等卖点进行详尽阐述的科普文章&#xff0c;动辄就…

计算机网络——理论知识总结(上)

开新番&#xff0c;因为博主备考的学校计网只考察1/6的分值&#xff0c;而且定位偏向于送分题&#xff0c;因此在备考时并没有很高强度的复习。本帖基于王道考研的教辅总结归纳&#xff0c;虽然是408的教材&#xff0c;但忽略其中有难度的部分&#xff0c;如计算题、画图题等&a…

如何通过员工工时管理降低企业成本?

作为当今快节奏商业环境的领导者或管理者&#xff0c;掌握员工的工作时间对于控制企业成本和确保每个人都各尽其责至关重要。 员工工时表软件就是这样一款工时跟踪管理解决方案&#xff1a;数字化的工时表有助于保护企业的财务不会被无节制的开支冲垮。然而&#xff0c;引入此…

【信创】银河麒麟V10 安装postgis

安装postGis步骤 1、安装 proj4 #tar -zxvf proj-4.8.0.tar.gz #cd proj-4.8.0 #mkdir -p /opt/proj-4.8.0 #./configure --prefix=/opt/proj-4.8.0 #make && make install #vi /etc/ld.so.conf.d/proj-4.8.0.conf #ldconfig 2、安装 geos #tar -xjf geos-3.6.1.tar.b…

Spark SQL概述与基本操作

目录 一、Spark SQL概述 &#xff08;1&#xff09;概念 &#xff08;2&#xff09;特点 &#xff08;3&#xff09;Spark SQL与Hive异同 &#xff08;4&#xff09;Spark的数据抽象 二、Spark Session对象执行环境构建 (1)Spark Session对象 &#xff08;2&#xff09;代码演…

hive使用中的参数优化与问题排查

1.使用hive的虚拟列排查错误案例 set hive.exec.rowoffsettrue; SELECT –输入文件名 INPUT__FILE__NAME, –文件中的块内偏移量 BLOCK__OFFSET__INSIDE__FILE, –文件行偏移量 ROW__OFFSET__INSIDE__BLOCK, * from hdp_lbg_zhaopin_defaultdb.zzdetail where dt‘20201117’…

07-定位布局

定位布局 1.定位布局- 定位流分类1.1.静态定位1.2.相对定位1.3.绝对定位1.4.固定定位1.5.粘滞定位1.6.z-index - 1.1.静态定位&#xff08; Static positioning&#xff09;- 1.2.什么是相对定位?&#xff08; Relative positioning &#xff09;- 相对定位注意点- 相对定位应…

Flink on yarn 加载失败plugins失效问题解决

Flink on yarn 加载失败plugins失效问题解决 flink版本&#xff1a;1.13.6 1. 问题 flink 任务运行在yarn集群,plugins加载失效,导致通过扩展资源获取任务参数失效 2. 问题定位 yarn容器的jar包及插件信息,jar包是正常上传 源码定位 加载plugins入口&#xff0c;TaskMana…

Unity的屏幕坐标获取

Screen.width public static int width ; 描述 屏幕窗口的当前宽度(以像素为单位)(只读)。 此为玩家窗口的实际宽度(在全屏模式下,它也是当前分辨率)。 using System.Collections; using System.Collections.Generic; using UnityEngine;public class Example : Mo…

TCP三次握手具体过程

四次挥手 1&#xff09;客户端进程发出连接释放报文&#xff0c;并且停止发送数据。释放数据报文首部&#xff0c;FIN1&#xff0c;其序列号为sequ&#xff08;等于前已经传送过来的数据的最后一个字节的序号加1)&#xff0c;此时&#xff0c;客户端进入FIN_WAIT_1&#xff08…

【Python机器学习】零基础掌握RandomTreesEmbedding集成学习

如何在高维数据中找到隐藏的结构? 面临大量复杂、高维的数据,例如社交网络分析、电子商务推荐系统或医疗诊断,如何有效地分析和解读这些数据成为一大挑战。一个有效的方法是使用嵌入技术将高维数据转化为低维形式,同时保留其内在结构。这次将介绍一种称为“随机树嵌入”(…

AI新能量!FortiGate NGFW面向数据中心全面集成FortiGuard AI 安全服务

企业IT技术正在以惊人的速度发展&#xff0c;转型最大的领域之一是下一代防火墙&#xff08;NGFW&#xff09;市场。如今&#xff0c;混合云、多云、边缘等多种基础设施形态共存&#xff0c;已经成为大部分企业的常态&#xff0c;不断扩张的攻击面需要不同形态防火墙的安全防护…

若依ruoyi-nbcio如何做一个仿钉钉流程设计器的思考

更多ruoyi-nbcio功能请看演示系统 gitee源代码地址 前后端代码&#xff1a; https://gitee.com/nbacheng/ruoyi-nbcio 演示地址&#xff1a;RuoYi-Nbcio后台管理系统 看到有些流程图采用仿钉钉的流程设计&#xff0c;比如下面界面&#xff1a; 这种方式虽然简单&#xff0c…

kafka安装配置

Kafka的安装配置可以按照以下步骤进行&#xff1a; 确保已安装Java运行环境&#xff1a;Kafka是使用Java语言编写的&#xff0c;因此需要在安装Kafka之前先安装Java运行环境。Kafka支持Java 8及以上版本。硬件要求&#xff1a;Kafka可以在任何硬件上运行&#xff0c;但是在生产…

计算机网络文章荟萃

脑残式网络编程入门(二)&#xff1a;我们在读写Socket时&#xff0c;究竟在读写什么&#xff1f;-网络编程/专项技术区 - 即时通讯开发者社区! 1.什么是 socket - 掘金2.socket 的实现原理 - 掘金本文讲述了 socket 在 linux 操作系统下的数据结构&#xff0c;以及阻塞 IO 利用…

【Java】PAT Basic Level 1023 组个最小数

题目 1024 组个最小数 作者 CAO, Peng 单位 Google 给定数字 0-9 各若干个。你可以以任意顺序排列这些数字&#xff0c;但必须全部使用。目标是使得最后得到的数尽可能小&#xff08;注意 0 不能做首位&#xff09;。例如&#xff1a;给定两个 0&#xff0c;两个 1&#xff…

檢測項目簡體字

某些項目可能要求代碼中不允許使用簡體字 安裝stcheck檢查 yarn add stcheck --dev在項目根目錄創建 st.config.json 文件 {"patterns": ["./**/*.(ts|js|tsx|jsx|vue|html)","!**/node_modules/**","!.git/**"],"gitignore&q…