-
Notifications
You must be signed in to change notification settings - Fork 1.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Introduce Flow.any, Flow.all, Flow.none
Fixes #4212
- Loading branch information
1 parent
ec83195
commit bed3d29
Showing
4 changed files
with
219 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
107 changes: 107 additions & 0 deletions
107
kotlinx-coroutines-core/common/src/flow/terminal/Logic.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,107 @@ | ||
@file:JvmMultifileClass | ||
@file:JvmName("FlowKt") | ||
|
||
package kotlinx.coroutines.flow | ||
|
||
import kotlinx.coroutines.* | ||
import kotlin.jvm.* | ||
|
||
|
||
/** | ||
* A terminal operator that returns `true` and immediately cancels the flow | ||
* if at least one element matches the given [predicate]. | ||
* | ||
* If the flow does not emit any elements or no element matches the predicate, the function returns `false`. | ||
* | ||
* Equivalent to `!all { !predicate(it) }` (see [Flow.all]) and `!none { predicate(it) }` (see [Flow.none]). | ||
* | ||
* Example: | ||
* | ||
* ``` | ||
* val myFlow = flow { | ||
* repeat(10) { | ||
* emit(it) | ||
* } | ||
* throw RuntimeException("You still didn't find the required number? I gave you ten!") | ||
* } | ||
* println(myFlow.any { it > 5 }) // true | ||
* println(flowOf(1, 2, 3).any { it > 5 }) // false | ||
* ``` | ||
* | ||
* @see Iterable.any | ||
* @see Sequence.any | ||
*/ | ||
public suspend fun <T> Flow<T>.any(predicate: suspend (T) -> Boolean): Boolean { | ||
var found = false | ||
collectWhile { | ||
val satisfies = predicate(it) | ||
if (satisfies) found = true | ||
!satisfies | ||
} | ||
return found | ||
} | ||
|
||
/** | ||
* A terminal operator that returns `true` if all elements match the given [predicate], | ||
* or returns `false` and cancels the flow as soon as the first element not matching the predicate is encountered. | ||
* | ||
* If the flow terminates without emitting any elements, the function returns `true` because there | ||
* are no elements in it that *do not* match the predicate. | ||
* See a more detailed explanation of this logic concept in the | ||
* ["Vacuous truth"](https://en.wikipedia.org/wiki/Vacuous_truth) article. | ||
* | ||
* Equivalent to `!any { !predicate(it) }` (see [Flow.any]) and `none { !predicate(it) }` (see [Flow.none]). | ||
* | ||
* Example: | ||
* | ||
* ``` | ||
* val myFlow = flow { | ||
* repeat(10) { | ||
* emit(it) | ||
* } | ||
* throw RuntimeException("You still didn't find the required number? I gave you ten!") | ||
* } | ||
* println(myFlow.all { it <= 5 }) // false | ||
* println(flowOf(1, 2, 3).all { it <= 5 }) // true | ||
* ``` | ||
* | ||
* @see Iterable.all | ||
* @see Sequence.all | ||
*/ | ||
public suspend fun <T> Flow<T>.all(predicate: suspend (T) -> Boolean): Boolean { | ||
var foundCounterExample = false | ||
collectWhile { | ||
val satisfies = predicate(it) | ||
if (!satisfies) foundCounterExample = true | ||
satisfies | ||
} | ||
return !foundCounterExample | ||
} | ||
|
||
/** | ||
* A terminal operator that returns `true` if no elements match the given [predicate], | ||
* or returns `false` and cancels the flow as soon as the first element matching the predicate is encountered. | ||
* | ||
* If the flow terminates without emitting any elements, the function returns `true` because there | ||
* are no elements in it that match the predicate. | ||
* See a more detailed explanation of this logic concept in the | ||
* ["Vacuous truth"](https://en.wikipedia.org/wiki/Vacuous_truth) article. | ||
* | ||
* Equivalent to `!any(predicate)` (see [Flow.any]) and `all { !predicate(it) }` (see [Flow.all]). | ||
* | ||
* Example: | ||
* ``` | ||
* val myFlow = flow { | ||
* repeat(10) { | ||
* emit(it) | ||
* } | ||
* throw RuntimeException("You still didn't find the required number? I gave you ten!") | ||
* } | ||
* println(myFlow.none { it > 5 }) // false | ||
* println(flowOf(1, 2, 3).none { it > 5 }) // true | ||
* ``` | ||
* | ||
* @see Iterable.none | ||
* @see Sequence.none | ||
*/ | ||
public suspend fun <T> Flow<T>.none(predicate: suspend (T) -> Boolean): Boolean = !any(predicate) |
106 changes: 106 additions & 0 deletions
106
kotlinx-coroutines-core/common/test/flow/operators/BooleanTerminationTest.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,106 @@ | ||
package kotlinx.coroutines.flow | ||
|
||
import kotlinx.coroutines.testing.* | ||
import kotlin.test.* | ||
|
||
class BooleanTerminationTest : TestBase() { | ||
@Test | ||
fun testAnyNominal() = runTest { | ||
val flow = flow { | ||
emit(1) | ||
emit(2) | ||
} | ||
|
||
assertTrue(flow.any { it > 0 }) | ||
assertTrue(flow.any { it % 2 == 0 }) | ||
assertFalse(flow.any { it > 5 }) | ||
} | ||
|
||
@Test | ||
fun testAnyEmpty() = runTest { | ||
assertFalse(emptyFlow<Int>().any { it > 0 }) | ||
} | ||
|
||
@Test | ||
fun testAnyInfinite() = runTest { | ||
assertTrue(flow { while (true) { emit(5) } }.any { it == 5 }) | ||
} | ||
|
||
@Test | ||
fun testAnyShortCircuit() = runTest { | ||
assertTrue(flow { | ||
emit(1) | ||
emit(2) | ||
expectUnreached() | ||
}.any { | ||
it == 2 | ||
}) | ||
} | ||
|
||
@Test | ||
fun testAllNominal() = runTest { | ||
val flow = flow { | ||
emit(1) | ||
emit(2) | ||
} | ||
|
||
assertTrue(flow.all { it > 0 }) | ||
assertFalse(flow.all { it % 2 == 0 }) | ||
assertFalse(flow.all { it > 5 }) | ||
} | ||
|
||
@Test | ||
fun testAllEmpty() = runTest { | ||
assertTrue(emptyFlow<Int>().all { it > 0 }) | ||
} | ||
|
||
@Test | ||
fun testAllInfinite() = runTest { | ||
assertFalse(flow { while (true) { emit(5) } }.all { it == 0 }) | ||
} | ||
|
||
@Test | ||
fun testAllShortCircuit() = runTest { | ||
assertFalse(flow { | ||
emit(1) | ||
emit(2) | ||
expectUnreached() | ||
}.all { | ||
it <= 1 | ||
}) | ||
} | ||
|
||
@Test | ||
fun testNoneNominal() = runTest { | ||
val flow = flow { | ||
emit(1) | ||
emit(2) | ||
} | ||
|
||
assertFalse(flow.none { it > 0 }) | ||
assertFalse(flow.none { it % 2 == 0 }) | ||
assertTrue(flow.none { it > 5 }) | ||
} | ||
|
||
@Test | ||
fun testNoneEmpty() = runTest { | ||
assertTrue(emptyFlow<Int>().none { it > 0 }) | ||
} | ||
|
||
@Test | ||
fun testNoneInfinite() = runTest { | ||
assertFalse(flow { while (true) { emit(5) } }.none { it == 5 }) | ||
} | ||
|
||
@Test | ||
fun testNoneShortCircuit() = runTest { | ||
assertFalse(flow { | ||
emit(1) | ||
emit(2) | ||
expectUnreached() | ||
}.none { | ||
it == 2 | ||
}) | ||
} | ||
|
||
} |