Kotlin - React Complete
This commit is contained in:
parent
11dbf563c2
commit
88fc9c2c3b
3 changed files with 296 additions and 0 deletions
22
kotlin/react/README.md
Normal file
22
kotlin/react/README.md
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
# React
|
||||||
|
|
||||||
|
Implement a basic reactive system.
|
||||||
|
|
||||||
|
Reactive programming is a programming paradigm that focuses on how values
|
||||||
|
are computed in terms of each other to allow a change to one value to
|
||||||
|
automatically propagate to other values, like in a spreadsheet.
|
||||||
|
|
||||||
|
Implement a basic reactive system with cells with settable values ("input"
|
||||||
|
cells) and cells with values computed in terms of other cells ("compute"
|
||||||
|
cells). Implement updates so that when an input value is changed, values
|
||||||
|
propagate to reach a new stable system state.
|
||||||
|
|
||||||
|
In addition, compute cells should allow for registering change notification
|
||||||
|
callbacks. Call a cell’s callbacks when the cell’s value in a new stable
|
||||||
|
state has changed from the previous stable state.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## Submitting Incomplete Solutions
|
||||||
|
It's possible to submit an incomplete solution so you can see how others have completed the exercise.
|
65
kotlin/react/src/main/kotlin/React.kt
Normal file
65
kotlin/react/src/main/kotlin/React.kt
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
class Reactor<T>() {
|
||||||
|
abstract inner class Cell {
|
||||||
|
abstract val value: T
|
||||||
|
internal val dependents = mutableListOf<ComputeCell>()
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Subscription {
|
||||||
|
fun cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
inner class InputCell(initialValue: T) : Cell() {
|
||||||
|
override var value: T = initialValue
|
||||||
|
set(newValue) {
|
||||||
|
field = newValue
|
||||||
|
dependents.forEach { it.propagate() }
|
||||||
|
dependents.forEach { it.fireCallbacks() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
inner class ComputeCell private constructor(val newValue: () -> T) : Cell() {
|
||||||
|
override var value: T = newValue()
|
||||||
|
private set
|
||||||
|
|
||||||
|
private var lastCallbackValue = value
|
||||||
|
private var callbacksIssued = 0
|
||||||
|
private val activeCallbacks = mutableMapOf<Int, (T) -> Any>()
|
||||||
|
|
||||||
|
constructor(vararg cells: Cell, f: (List<T>) -> T) : this({ f(cells.map { it.value }) }) {
|
||||||
|
for (cell in cells) {
|
||||||
|
cell.dependents.add(this)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun addCallback(f: (T) -> Any): Subscription {
|
||||||
|
val id = callbacksIssued
|
||||||
|
callbacksIssued++
|
||||||
|
activeCallbacks[id] = f
|
||||||
|
return object : Subscription {
|
||||||
|
override fun cancel() {
|
||||||
|
activeCallbacks.remove(id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal fun propagate() {
|
||||||
|
val nv = newValue()
|
||||||
|
if (nv == value) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
value = nv
|
||||||
|
dependents.forEach { it.propagate() }
|
||||||
|
}
|
||||||
|
|
||||||
|
internal fun fireCallbacks() {
|
||||||
|
if (value == lastCallbackValue) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
lastCallbackValue = value
|
||||||
|
for (cb in activeCallbacks.values) {
|
||||||
|
cb(value)
|
||||||
|
}
|
||||||
|
dependents.forEach { it.fireCallbacks() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
209
kotlin/react/src/test/kotlin/ReactTest.kt
Normal file
209
kotlin/react/src/test/kotlin/ReactTest.kt
Normal file
|
@ -0,0 +1,209 @@
|
||||||
|
import org.junit.Test
|
||||||
|
import org.junit.Ignore
|
||||||
|
import org.junit.runner.RunWith
|
||||||
|
import org.junit.runners.Parameterized
|
||||||
|
import kotlin.test.assertEquals
|
||||||
|
|
||||||
|
class ReactTest {
|
||||||
|
@Test
|
||||||
|
fun inputCellsHaveValue() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(10)
|
||||||
|
assertEquals(10, input.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun inputCellsValueCanBeSet() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(4)
|
||||||
|
input.value = 20
|
||||||
|
assertEquals(20, input.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun computeCellsCalculateInitialValue() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val output = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
assertEquals(2, output.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun computeCellsTakeInputsInTheRightOrder() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val one = reactor.InputCell(1)
|
||||||
|
val two = reactor.InputCell(2)
|
||||||
|
val output = reactor.ComputeCell(one, two) { (x, y) -> x + y * 10 }
|
||||||
|
assertEquals(21, output.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun computeCellsUpdateValueWhenDependenciesAreChanged() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val output = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
input.value = 3
|
||||||
|
assertEquals(4, output.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun computeCellsCanDependOnOtherComputeCells() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val timesTwo = reactor.ComputeCell(input) { it[0] * 2 }
|
||||||
|
val timesThirty = reactor.ComputeCell(input) { it[0] * 30 }
|
||||||
|
val output = reactor.ComputeCell(timesTwo, timesThirty) { (x, y) -> x + y }
|
||||||
|
|
||||||
|
assertEquals(32, output.value)
|
||||||
|
input.value = 3
|
||||||
|
assertEquals(96, output.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun computeCellsFireCallbacks() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val output = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
|
||||||
|
val vals = mutableListOf<Int>()
|
||||||
|
output.addCallback { vals.add(it) }
|
||||||
|
|
||||||
|
input.value = 3
|
||||||
|
assertEquals(listOf(4), vals)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun callbacksOnlyFireOnChange() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val output = reactor.ComputeCell(input) { if (it[0] < 3) 111 else 222 }
|
||||||
|
|
||||||
|
val vals = mutableListOf<Int>()
|
||||||
|
output.addCallback { vals.add(it) }
|
||||||
|
|
||||||
|
input.value = 2
|
||||||
|
assertEquals(listOf<Int>(), vals)
|
||||||
|
|
||||||
|
input.value = 4
|
||||||
|
assertEquals(listOf(222), vals)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun callbacksCanBeAddedAndRemoved() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(11)
|
||||||
|
val output = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
|
||||||
|
val vals1 = mutableListOf<Int>()
|
||||||
|
val sub1 = output.addCallback { vals1.add(it) }
|
||||||
|
val vals2 = mutableListOf<Int>()
|
||||||
|
output.addCallback { vals2.add(it) }
|
||||||
|
|
||||||
|
input.value = 31
|
||||||
|
sub1.cancel()
|
||||||
|
|
||||||
|
val vals3 = mutableListOf<Int>()
|
||||||
|
output.addCallback { vals3.add(it) }
|
||||||
|
|
||||||
|
input.value = 41
|
||||||
|
|
||||||
|
assertEquals(listOf(32), vals1)
|
||||||
|
assertEquals(listOf(32, 42), vals2)
|
||||||
|
assertEquals(listOf(42), vals3)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun removingACallbackMultipleTimesDoesntInterfereWithOtherCallbacks() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val output = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
|
||||||
|
val vals1 = mutableListOf<Int>()
|
||||||
|
val sub1 = output.addCallback { vals1.add(it) }
|
||||||
|
val vals2 = mutableListOf<Int>()
|
||||||
|
output.addCallback { vals2.add(it) }
|
||||||
|
|
||||||
|
for (i in 1..10) {
|
||||||
|
sub1.cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
input.value = 2
|
||||||
|
assertEquals(listOf<Int>(), vals1)
|
||||||
|
assertEquals(listOf(3), vals2)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun callbacksShouldOnlyBeCalledOnceEvenIfMultipleDependenciesChange() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val plusOne = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
val minusOne1 = reactor.ComputeCell(input) { it[0] - 1 }
|
||||||
|
val minusOne2 = reactor.ComputeCell(minusOne1) { it[0] - 1 }
|
||||||
|
val output = reactor.ComputeCell(plusOne, minusOne2) { (x, y) -> x * y }
|
||||||
|
|
||||||
|
val vals = mutableListOf<Int>()
|
||||||
|
output.addCallback { vals.add(it) }
|
||||||
|
|
||||||
|
input.value = 4
|
||||||
|
assertEquals(listOf(10), vals)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun callbacksShouldNotBeCalledIfDependenciesChangeButOutputValueDoesntChange() {
|
||||||
|
val reactor = Reactor<Int>()
|
||||||
|
val input = reactor.InputCell(1)
|
||||||
|
val plusOne = reactor.ComputeCell(input) { it[0] + 1 }
|
||||||
|
val minusOne = reactor.ComputeCell(input) { it[0] - 1 }
|
||||||
|
val alwaysTwo = reactor.ComputeCell(plusOne, minusOne) { (x, y) -> x - y }
|
||||||
|
|
||||||
|
val vals = mutableListOf<Int>()
|
||||||
|
alwaysTwo.addCallback { vals.add(it) }
|
||||||
|
|
||||||
|
for (i in 1..10) {
|
||||||
|
input.value = i
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEquals(listOf<Int>(), vals)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@RunWith(Parameterized::class)
|
||||||
|
// This is a digital logic circuit called an adder:
|
||||||
|
// https://en.wikipedia.org/wiki/Adder_(electronics)
|
||||||
|
class ReactAdderTest(val input: Input, val expected: Expected) {
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
data class Input(val a: Boolean, val b: Boolean, val carryIn: Boolean)
|
||||||
|
data class Expected(val carryOut: Boolean, val sum: Boolean)
|
||||||
|
|
||||||
|
@JvmStatic
|
||||||
|
@Parameterized.Parameters(name = "{index}: {0} = {1}")
|
||||||
|
fun data() = listOf(
|
||||||
|
arrayOf(Input(a=false, b=false, carryIn=false), Expected(carryOut=false, sum=false)),
|
||||||
|
arrayOf(Input(a=false, b=false, carryIn=true), Expected(carryOut=false, sum=true)),
|
||||||
|
arrayOf(Input(a=false, b=true, carryIn=false), Expected(carryOut=false, sum=true)),
|
||||||
|
arrayOf(Input(a=false, b=true, carryIn=true), Expected(carryOut=true, sum=false)),
|
||||||
|
arrayOf(Input(a=true, b=false, carryIn=false), Expected(carryOut=false, sum=true)),
|
||||||
|
arrayOf(Input(a=true, b=false, carryIn=true), Expected(carryOut=true, sum=false)),
|
||||||
|
arrayOf(Input(a=true, b=true, carryIn=false), Expected(carryOut=true, sum=false)),
|
||||||
|
arrayOf(Input(a=true, b=true, carryIn=true), Expected(carryOut=true, sum=true))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun test() {
|
||||||
|
val reactor = Reactor<Boolean>()
|
||||||
|
val a = reactor.InputCell(input.a)
|
||||||
|
val b = reactor.InputCell(input.b)
|
||||||
|
val carryIn = reactor.InputCell(input.carryIn)
|
||||||
|
|
||||||
|
val aXorB = reactor.ComputeCell(a, b) { (x, y) -> x.xor(y) }
|
||||||
|
val sum = reactor.ComputeCell(aXorB, carryIn) { (x, y) -> x.xor(y) }
|
||||||
|
|
||||||
|
val aXorBAndCin = reactor.ComputeCell(aXorB, carryIn) { (x, y) -> x && y }
|
||||||
|
val aAndB = reactor.ComputeCell(a, b) { (x, y) -> x && y }
|
||||||
|
val carryOut = reactor.ComputeCell(aXorBAndCin, aAndB) { (x, y) -> x || y }
|
||||||
|
|
||||||
|
assertEquals(expected, Expected(sum=sum.value, carryOut=carryOut.value))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue