- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
Kotlin 中是否有一种方法可以“窥视”迭代器的下一个元素而不推进它?对于示例用例,请考虑此函数用于合并两个预排序的序列:
fun merge(seq1: Sequence<Int>, seq2: Sequence<Int>) = sequence<Int> {
val it1 = seq1.iterator()
var current1 = if (it1.hasNext()) it1.next() else null
val it2 = seq2.iterator()
var current2 = if (it2.hasNext()) it2.next() else null
while (current1 != null && current2 != null) {
if (current1 <= current2) {
yield(current1)
current1 = if (it1.hasNext()) it1.next() else null
} else {
yield(current2)
current2 = if (it2.hasNext()) it2.next() else null
}
}
while (current1 != null) {
yield(current1)
current1 = if (it1.hasNext()) it1.next() else null
}
while (current2 != null) {
yield(current2)
current2 = if (it2.hasNext()) it2.next() else null
}
}
这个函数必须跳过重重困难,因为Iterator
接口(interface)无法在不前进的情况下请求下一个元素。换句话说,它不尊重 Command-Query Separation Principle .
如果我使用 Guava 的 PeekingIterator ,我可以使这个函数更加简洁和可读。 :
fun merge(seq1: Sequence<Int>, seq2: Sequence<Int>) = sequence<Int> {
val it1: PeekingIterator<Int> = Iterators.peekingIterator(seq1.iterator())
val it2: PeekingIterator<Int> = Iterators.peekingIterator(seq2.iterator())
while (it1.hasNext() && it2.hasNext())
yield((if (it1.peek() <= it2.peek()) it1 else it2).next())
yieldAll(it1)
yieldAll(it2)
}
相同的功能从 24 行减少到 9 行!然而,我在 Kotlin 项目中使用 Guava 感觉不太舒服。 Guava 的创建是为了解决 Java API 的一些限制。 Kotlin 标准库要丰富得多,我们不需要使用第三方 Java 库。有没有办法在不使用 Guava 的情况下使第一个代码示例更加简洁和可读?
附录
这是上述函数的示例单元测试,它应该让您了解它的行为方式:
import io.kotest.core.spec.style.FreeSpec
import io.kotest.matchers.shouldBe
class MyTest : FreeSpec({
"merge test" {
merge(emptySequence(), emptySequence()).toList() shouldBe emptyList()
merge(emptySequence(), sequenceOf(1)).toList() shouldBe listOf(1)
merge(emptySequence(), sequenceOf(1, 3)).toList() shouldBe listOf(1, 3)
merge(sequenceOf(1), emptySequence()).toList() shouldBe listOf(1)
merge(sequenceOf(1, 3), emptySequence()).toList() shouldBe listOf(1, 3)
merge(sequenceOf(0, 3, 4), sequenceOf(1, 4, 7)).toList() shouldBe listOf(0, 1, 3, 4, 4, 7)
}
)
最佳答案
我认为 Kotlin stdlib 没有提供这个,但我们可以自己实现:
fun main() {
val iter = listOf(1, 2, 3).iterator().peeking()
println(iter.next()) // 1
println(iter.peek()) // 2
println(iter.peek()) // 2
println(iter.next()) // 2
println(iter.peek()) // 3
println(iter.next()) // 3
println(iter.peek()) // NoSuchElementException
}
fun <T> Iterator<T>.peeking() = object : PeekingIterator<T> {
private var hasPeeked = false
private var peeked: T? = null
override fun hasNext(): Boolean = hasPeeked || <a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="2e5a46475d6e5e4b4b4547404900464f5d604b565a" rel="noreferrer noopener nofollow">[email protected]</a>()
override fun next(): T {
return if (hasPeeked) {
hasPeeked = false
@Suppress("UNCHECKED_CAST")
peeked as T
} else {
<a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="691d01001a29190c0c0200070e47070c111d" rel="noreferrer noopener nofollow">[email protected]</a>()
}
}
override fun peek(): T {
if (!hasPeeked) {
peeked = <a href="https://stackoverflow.com/cdn-cgi/l/email-protection" class="__cf_email__" data-cfemail="a1d5c9c8d2e1d1c4c4cac8cfc68fcfc4d9d5" rel="noreferrer noopener nofollow">[email protected]</a>()
hasPeeked = true
}
@Suppress("UNCHECKED_CAST")
return peeked as T
}
}
interface PeekingIterator<T> : Iterator<T> {
fun peek(): T
}
关于kotlin - 窥探迭代器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/77174295/
我正在查看Kotlin Github page我注意到 Kotlin 语言本身大部分是用 Kotlin 编写的:我只是想知道,一种语言怎么可能大部分都是用它自己的语言编写的?在您可以使用正在创建的语言
我有以下非常简单的 kotlin 代码来演示中缀函数 com.lopushen.demo.presentation 包 fun main(args: Array) { print("Hello
我在 Java 中有 2 个模型类,其中一个扩展了另一个 @UseStag public class GenericMessages extends NavigationLocalizationMap
Kotlin 代码 runBlocking { flow { for (i in 0..4) { println("Emit $i")
这三个 Kotlin 插件和它们的实际作用有什么区别? plugins { id 'kotlin-android' id 'org.jetbrains.kotlin.android'
我正在为某些现有库添加 Kotlin 原生 linuxX64 目标支持。库已成功编译,但在运行测试用例时,出现以下运行时错误: kotlin.native.concurrent.InvalidMuta
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题吗? 通过 editing this post 添加细节并澄清问题. 关闭 2 年前。 Improve t
我创建了一个类并向其添加了一个与成员函数具有相同签名的扩展,并执行了这个方法,它总是执行成员方法。 class Worker { fun work() = "...working" } fun
我知道传递给函数的参数将被视为“val”,即使变量被初始化为“var”。但这对我来说一直是个问题。在下面的示例代码中,我想通过使用函数“changeNum”修改变量“num”的值。但当然,Kotlin
现在,我正在尝试用 Kotlin 重写我的 Java 应用程序。然后,我遇到了日志语句,比如 log.info("do the print thing for {}", arg); 所以我有两种方法可
有点出名article关于许多语言的异步编程模型的状态,指出它们存在“颜色”问题,特别是将生态系统分为两个独立的世界:异步和非异步。以下是这种语言的属性: 每个函数都有一种颜色,红色或蓝色(例如asy
因为 KDoc 文档生成引擎是 abandoned in favor of Dokka , Kotlin 文档应该称为“KDoc 注释”,还是“Dokka 注释”? 最佳答案 如所述here , KD
我想在可空对象上传递函数引用。以 Android 为例,假设我想使用 Activity#onBackPressed来自作为该事件的子级的片段。 如果我想调用这个函数,我可以很容易地做到 activit
我有一个列表 (x, y)其中y只能是 0 或 1 这样 例如: [(3, 0), (3, 1), (5, 1)] [(5, 0), (3, 1), (5, 1)] [(1, 1), (3, 1),
从强类型语言的定义来看: A strongly-typed programming language is one in which each type of data (such as intege
这不能编译的事实是否意味着它们不是一流的类型? fun foo(s: String): Int = s.length // This won't compile. val bar = foo 有没有办
如果在 Java i++是一个表达式和 i++;是一个表达式语句,分号(;) 在 Kotlin 中是可选的,是 i++ Kotlin 中的表达式或表达式语句? 最佳答案 i++是一个表达式,因为它有一
代码(如下所示)是否正确?它取自 Kotlin-docs.pdf 的第 63 页,这也是 https://kotlinlang.org/docs/reference/generics.html 的最后
我正在尝试使用 Kotlin 为 Android 的一些全局 API 解析器(检查网络连接、调用 API 并通过来自源的单个调用返回格式化数据),并且在某些时候我不得不创建一个通用类型 object就
kotlinlang 中的任务: 使用月份变量重写此模式,使其与格式 13 JUN 1992(两位数字、一个空格、一个月份缩写、一个空格、四位数字)中的日期相匹配。 答案是:val month = "
我是一名优秀的程序员,十分优秀!