1
1
Fork 0

Regenerated samples

This commit is contained in:
Svetlana Isakova 2020-11-17 20:00:10 +01:00
parent f5ea6e73af
commit a3d7a6660a
75 changed files with 617 additions and 446 deletions

View File

@ -1,25 +1,28 @@
// Interoperability/BigFibonacci.kt
package interop
import atomictest.eq
import bigint.*
import java.math.BigInteger
import java.math.BigInteger.ONE
import java.math.BigInteger.ZERO
fun fibonacci(n: Int): BigInt {
fun fibonacci(n: Int): BigInteger {
tailrec fun fibonacci(
n: Int,
current: BigInt,
next: BigInt
): BigInt {
current: BigInteger,
next: BigInteger
): BigInteger {
if (n == 0) return current
return fibonacci(
n - 1, next, current + next)
n - 1, next, current + next) // [1]
}
return fibonacci(n, zero, one)
return fibonacci(n, ZERO, ONE)
}
fun main() {
(0..7).map { fibonacci(it) } eq
"[0, 1, 1, 2, 3, 5, 8, 13]"
fibonacci(22) eq 17711.big
fibonacci(22) eq 17711.toBigInteger()
fibonacci(150) eq
"9969216677189303386214405760200".big
"9969216677189303386214405760200"
.toBigInteger()
}

View File

@ -1,14 +1,13 @@
// Interoperability/BigInt.kt
package bigint
package biginteger
import java.math.BigInteger
typealias BigInt = BigInteger
fun Int.toBigInteger(): BigInteger =
BigInteger.valueOf(this.toLong())
val Int.big: BigInt
get() = BigInt.valueOf(toLong())
fun String.toBigInteger(): BigInteger =
BigInteger(this)
val String.big: BigInt
get() = BigInt(this)
val zero = BigInt.ZERO
val one = BigInt.ONE
operator fun BigInteger.plus(
other: BigInteger
): BigInteger = this.add(other)

View File

@ -2,8 +2,8 @@
package Interoperability;
import java.io.Serializable;
public class
Chameleon implements Serializable {
public class Chameleon
implements Serializable {
private int size;
private String color;
public int getSize() {

View File

@ -8,7 +8,7 @@ public class JavaChecked {
// Build path to current source file, based
// on directory where Gradle is invoked:
static Path thisFile = Paths.get(
"DataFiles", "file_real.txt");
"DataFiles", "file_wubba.txt");
public static void main(String[] args) {
BufferedReader source = null;
try {

View File

@ -0,0 +1,13 @@
// Interoperability/JavaWrapper.java
package Interoperability;
import java.util.List;
public class JavaWrapper {
public static void main(String[] args) {
// Primitive type
int i = 10;
// Wrapper types
Integer iOrNull = null;
List<Integer> list = List.of(1, 2, 3);
}
}

View File

@ -0,0 +1,25 @@
// Interoperability/UseDataClass.java
package Interoperability;
import java.util.HashMap;
import static atomictest.AtomicTestKt.eq;
public class UseDataClass {
public static void main(String[] args) {
Muppet m = new Muppet("Ernie", 6);
int ageErnie = m.getAge();
m.setName("Bert");
m.setAge(7);
eq(ageErnie < m.getAge(), true);
eq(m, "Muppet(name=Bert, age=7)");
// Call copy() from the data class:
Muppet mc = m.copy("???", 5);
eq(mc, "Muppet(name=???, age=5)");
HashMap<Muppet, String> hm =
new HashMap<>();
// Muppets work as hash keys:
hm.put(m, "Happy Muppet");
eq(hm.get(m), "Happy Muppet");
}
}

View File

@ -1,24 +0,0 @@
// Interoperability/UsingDataClass.java
package Interoperability;
import java.util.HashMap;
import static atomictest.AtomicTestKt.eq;
public class UsingDataClass {
public static void main(String[] args) {
Data d = new Data("Alice", 22);
String name = d.getName();
d.setName("Bob");
int age = d.getAge();
d.setAge(age + 1);
// toString():
eq(d, "Data(name=Bob, age=23)");
HashMap<Data, Integer> hm =
new HashMap<>();
hm.put(d, 47);
// Data objects work as hash keys:
eq((double)hm.get(d), (double)47);
// Call copy() from the Data class:
Data d2 = d.copy("Sam", 24);
eq(d2, "Data(name=Sam, age=24)");
}
}

View File

@ -3,7 +3,7 @@ import atomictest.eq
import java.io.File
fun main() {
File("DataFiles/file_real.txt")
File("DataFiles/file_wubba.txt")
.readLines()[0] eq
"wubba lubba dub dub"
}

View File

@ -1,7 +1,7 @@
// Interoperability/KotlinDataClass.kt
package Interoperability
data class Data(
data class Muppet(
var name: String,
var age: Int
)

View File

@ -0,0 +1,10 @@
// Interoperability/KotlinWrapper.kt
package interop
fun main() {
// Generates a primitive int:
val i = 10
// Generates wrapper types:
val iOrNull: Int? = null
val list: List<Int> = listOf(1, 2, 3)
}

View File

@ -14,7 +14,7 @@ files:
visible: true
- name: src/KotlinDataClass.kt
visible: true
- name: src/Interoperability/UsingDataClass.java
- name: src/Interoperability/UseDataClass.java
visible: true
- name: src/TopLevelFunction.kt
visible: true
@ -58,3 +58,7 @@ files:
visible: true
- name: src/ReadOnlyCollections.kt
visible: true
- name: src/Interoperability/JavaWrapper.java
visible: true
- name: src/KotlinWrapper.kt
visible: true

View File

@ -1 +0,0 @@
wubba lubba dub dub

View File

@ -1,9 +1,7 @@
## Maps (#3)
If you commonly select elements from a `List` using keys, it makes sense to
optimize by using a `Map` instead of a `List`. A `Map` finds an element in
constant time. With a `List`, in the worst case you must iterate over every
element.
optimize by using a `Map` instead of a `List`.
Change the internal implementation of the `Cage` class to store elements in a
`Map` rather than a `List`. To get an element use the `getValue()` member

View File

@ -1,6 +1,6 @@
## Summary 2 (#3)
Create a `FixedHolder` class with a constructor parameter that sets the
Create a `FixedSizeHolder` class with a constructor parameter that sets the
maximum number of `Any` objects that can be held. Add read-only `size` and
`full` properties. If the user calls `add()` when it's `full`, throw an
`IllegalStateException`.

View File

@ -11,9 +11,9 @@ example, `IV` contains two digits) and store the maximum value found so far. If
the next Roman digit is greater than or equal to the current maximum value, add
it to the result. If it's less than the maximum, subtract it instead. For
example, to convert `XLIV = 44`, iterate over `VILX` which is the reverse of
`XLIV`. You add `V`(`5`) and `L`(`50`), but subtract `1`(`I`) because it's less
than the current maximum `V`, and subtract `10`(`X`) because it's less than the
updated maximum `X`:
`XLIV`. You add `5`(`V`) and subtract `1`(`I`) because it's less than the
current maximum `5`(`V`), then and `50`(`L`) and subtract `10`(`X`) because it's
less than the updated maximum `10`(`X`):
| numeral | current maximum | action |
| ------- |-----------------|--------|

View File

@ -2,18 +2,13 @@
package extensionlambdas
import atomictest.eq
val divideBy = fun Int.(d: Int): Boolean {
return this % d == 0
}
fun exec(
arg1: Int, arg2: Int,
f: Int.(Int) -> Boolean
arg1: Int, arg2: Int,
f: Int.(Int) -> Boolean
) = arg1.f(arg2)
fun main() {
10.divideBy(2) eq true
10.divideBy(3) eq false
exec(10, 2, divideBy) eq true
exec(10, 3, divideBy) eq false
exec(10, 2, fun Int.(d: Int): Boolean {
return this % d == 0
}) eq true
}

View File

@ -13,7 +13,7 @@ fun lambdaUnitReturn () {
"So it can be anything ..."
}
unitReturn { 1 } // ... of any type ...
unitReturn { } // ... or nothing
unitReturn { } // ... or nothing
nonUnitReturn {
"Must return the proper type"
}

View File

@ -14,6 +14,10 @@ private fun clean() = buildString {
('a'..'z').forEach { append(it) }
}
private fun cleaner() =
"ABCs: " + ('a'..'z').fold("", String::plus)
fun main() {
messy() eq clean()
clean() eq cleaner()
}

View File

@ -2,6 +2,14 @@
package extensionlambdas
import atomictest.eq
fun String.transform1(
n: Int, lambda: (String, Int) -> String
) = lambda(this, n)
fun String.transform2(
n: Int, lambda: String.(Int) -> String
) = lambda(this, n)
val duplicate: String.(Int) -> String = {
repeat(it)
}
@ -12,14 +20,6 @@ val alternate: String.(Int) -> String = {
.joinToString("")
}
fun String.transform1(
n: Int, lambda: (String, Int) -> String
) = lambda(this, n)
fun String.transform2(
n: Int, lambda: String.(Int) -> String
) = lambda(this, n)
fun main() {
"hello".transform1(5, duplicate)
.transform2(3, alternate) eq "hleolhleo"

View File

@ -1,76 +0,0 @@
// OperatorOverloading/AllOperators.kt
package operatoroverloading
class C {
// equals() cannot be an extension
// function and must be defined specially:
override operator
fun equals(other: Any?) = true
}
// Unary operators:
operator fun C.unaryPlus() = this
operator fun C.unaryMinus() = this
operator fun C.not() = this
// Increment/decrement:
operator fun C.inc() = this
operator fun C.dec() = this
// Binary operators:
operator fun C.plus(rv: C) = rv
operator fun C.minus(rv: C) = rv
operator fun C.times(rv: C) = rv
operator fun C.div(rv: C) = rv
operator fun C.rem(rv: C) = rv
operator fun C.rangeTo(rv: C) = rv
// 'In' operator:
operator fun C.contains(rv: C) = true
// Indexed access:
operator fun C.get(i: Int) = this
operator fun C.set(i: Int, c: C) = this
// Augmented assignment:
operator fun C.plusAssign(rv: C) = Unit
operator fun C.minusAssign(rv: C) = Unit
operator fun C.timesAssign(rv: C) = Unit
operator fun C.divAssign(rv: C) = Unit
operator fun C.remAssign(rv: C) = Unit
// Comparison, must return Int:
operator fun C.compareTo(rv: C) = 1
fun main() {
val c = C()
+c // unaryPlus()
-c // unaryMinus()
!c // not()
var cc = C()
cc++ // inc() (needs var)
cc-- // dec() (needs var)
c + c // plus()
c - c // minus()
c * c // times()
c / c // div()
c % c // rem()
c..c // rangeTo()
c in c // contains()
c !in c // contains()
c[2] // get()
c[2] = c // set()
c += c // plusAssign()
c -= c // minusAssign()
c *= c // timesAssign()
c /= c // divAssign()
c %= c // remAssign()
// equals():
c == c
c != c
// compareTo():
c < c
c > c
c <= c
c >= c
}

View File

@ -0,0 +1,62 @@
// OperatorOverloading/ArithmeticOperators.kt
package operatoroverloading
import atomictest.eq
// Unary operators:
operator fun E.unaryPlus() = E(v)
operator fun E.unaryMinus() = E(-v)
operator fun E.not() = this
// Increment/decrement:
operator fun E.inc() = E(v + 1)
operator fun E.dec() = E(v - 1)
fun unary(a: E) {
+a // unaryPlus()
-a // unaryMinus()
!a // not()
var b = a
b++ // inc() (must be var)
b-- // dec() (must be var)
}
// Binary operators:
operator fun E.plus(e: E) = E(v + e.v)
operator fun E.minus(e: E) = E(v - e.v)
operator fun E.times(e: E) = E(v * e.v)
operator fun E.div(e: E) = E(v % e.v)
operator fun E.rem(e: E) = E(v / e.v)
fun binary(a: E, b: E) {
a + b // a.plus(b)
a - b // a.minus(b)
a * b // a.times(b)
a / b // a.div(b)
a % b // a.rem(b)
}
// Augmented assignment:
operator fun E.plusAssign(e: E) { v += e.v }
operator fun E.minusAssign(e: E) { v - e.v }
operator fun E.timesAssign(e: E) { v *= e.v }
operator fun E.divAssign(e: E) { v /= e.v }
operator fun E.remAssign(e: E) { v %= e.v }
fun assignment(a: E, b: E) {
a += b // a.plusAssign(b)
a -= b // a.minusAssign(b)
a *= b // a.timesAssign(b)
a /= b // a.divAssign(b)
a %= b // a.remAssign(b)
}
fun main() {
val a = E(2)
val b = E(3)
a + b eq E(5)
a * b eq E(6)
val x = E(1)
x += b * b
x eq E(10)
}

View File

@ -0,0 +1,15 @@
// OperatorOverloading/Comparison.kt
package operatoroverloading
import atomictest.eq
operator fun E.compareTo(e: E): Int =
v.compareTo(e.v)
fun main() {
val a = E(2)
val b = E(3)
(a < b) eq true // a.compareTo(b) < 0
(a > b) eq false // a.compareTo(b) > 0
(a <= b) eq true // a.compareTo(b) <= 0
(a >= b) eq false // a.compareTo(b) >= 0
}

View File

@ -0,0 +1,12 @@
// OperatorOverloading/ConfusingPrecedence.kt
package operatoroverloading
import atomictest.eq
fun main() {
val x: Int? = 1
val y: Int = 2
val sum = x ?: 0 + y
sum eq 1
(x ?: 0) + y eq 3 // [1]
x ?: (0 + y) eq 1 // [2]
}

View File

@ -0,0 +1,24 @@
// OperatorOverloading/ContainerAccess.kt
package operatoroverloading
import atomictest.eq
data class C(val c: MutableList<Int>) {
override fun toString() = "C($c)"
}
operator fun C.contains(e: E) = e.v in c
operator fun C.get(i: Int): E = E(c[i])
operator fun C.set(i: Int, e: E) {
c[i] = e.v
}
fun main() {
val c = C(mutableListOf(2, 3))
(E(2) in c) eq true // c.contains(E(2))
(E(4) in c) eq false // c.contains(E(4))
c[1] eq E(3) // c.get(1)
c[1] = E(4) // c.set(2, E(4))
c eq C(mutableListOf(2, 4))
}

View File

@ -0,0 +1,19 @@
// OperatorOverloading/DefaultEquality.kt
package operatoroverloading
import atomictest.eq
class A(val i: Int)
data class D(val i: Int)
fun main() {
// Normal class:
val a = A(1)
val b = A(1)
val c = a
(a == b) eq false
(a == c) eq true
// Data class:
val d = D(1)
val e = D(1)
(d == e) eq true
}

View File

@ -0,0 +1,22 @@
// OperatorOverloading/DefiningEquality.kt
package operatoroverloading
import atomictest.eq
class E(var v: Int) {
override fun equals(other: Any?) = when {
this === other -> true // [1]
other !is E -> false // [2]
else -> v == other.v // [3]
}
override fun hashCode(): Int = v
override fun toString() = "E($v)"
}
fun main() {
val a = E(1)
val b = E(2)
(a == b) eq false // a.equals(b)
(a != b) eq true // !a.equals(b)
// Reference equality:
(E(1) === E(1)) eq false
}

View File

@ -0,0 +1,9 @@
// OperatorOverloading/DifferentTypes.kt
package operatoroverloading
import atomictest.eq
operator fun E.plus(i: Int) = E(v + i)
fun main() {
E(1) + 10 eq E(11)
}

View File

@ -0,0 +1,24 @@
// OperatorOverloading/EqualsForNullable.kt
package operatoroverloading
import atomictest.eq
fun equalsWithIf(a: E?, b: E?) =
if (a === null)
b === null
else
a == b
fun equalsWithElvis(a: E?, b: E?) =
a?.equals(b) ?: (b === null)
fun main() {
val x: E? = null
val y = E(0)
val z: E? = null
(x == y) eq false
(x == z) eq true
equalsWithIf(x, y) eq false
equalsWithIf(x, z) eq true
equalsWithElvis(x, y) eq false
equalsWithElvis(x, z) eq true
}

View File

@ -0,0 +1,21 @@
// OperatorOverloading/Ranges.kt
package operatoroverloading
import atomictest.eq
data class R(val r: IntRange) { // Range
override fun toString() = "R($r)"
}
operator fun E.rangeTo(e: E) = R(v..e.v)
operator fun R.contains(e: E): Boolean =
e.v in r
fun main() {
val a = E(2)
val b = E(3)
val r = a..b // a.rangeTo(b)
(a in r) eq true // r.contains(a)
(a !in r) eq false // !r.contains(a)
r eq R(2..3)
}

View File

@ -6,8 +6,6 @@ files:
visible: true
- name: src/Molecule.kt
visible: true
- name: src/AllOperators.kt
visible: true
- name: src/Invoke.kt
visible: true
- name: src/StringInvoke.kt
@ -16,3 +14,21 @@ files:
visible: true
- name: src/Swearing.kt
visible: true
- name: src/DifferentTypes.kt
visible: true
- name: src/Ranges.kt
visible: true
- name: src/ContainerAccess.kt
visible: true
- name: src/DefiningEquality.kt
visible: true
- name: src/ArithmeticOperators.kt
visible: true
- name: src/ConfusingPrecedence.kt
visible: true
- name: src/EqualsForNullable.kt
visible: true
- name: src/Comparison.kt
visible: true
- name: src/DefaultEquality.kt
visible: true

View File

@ -1,9 +1,9 @@
// PropertyDelegation/CarService.kt
// DelegationTools/CarService.kt
package propertydelegation
import atomictest.eq
class Driver(
val map: MutableMap<String, Any?>
map: MutableMap<String, Any?>
) {
var name: String by map
var age: Int by map
@ -20,7 +20,7 @@ fun main() {
"available" to false,
"coord" to Pair(111.93, 1231.12)
)
val driver = Driver(info);
val driver = Driver(info)
driver.available eq false
driver.available = true
info eq "{name=Bruno Fiat, age=22, " +

View File

@ -1,75 +0,0 @@
// DelegationTools/DelegateProvider.kt
package delegationtools
import kotlin.reflect.KProperty
import atomictest.*
class Delegate<T>(private var value: T) {
operator fun getValue(
thisRef: Any?,
property: KProperty<*>
): T {
trace("getValue: $value")
return value
}
operator fun setValue(
thisRef: Any?,
property: KProperty<*>,
newValue: T
) {
trace("setValue: $newValue")
value = newValue
}
}
class DelegateProvider<T>(val value: T) {
operator fun provideDelegate(
thisRef: Any?,
property: KProperty<*>
): Delegate<T> {
trace("providing Delegate($value)")
return Delegate(value)
}
}
fun <T> delegate(value: T):
DelegateProvider<T> {
trace("creating DelegateProvider($value)")
return DelegateProvider(value)
}
fun main() {
var x by delegate(3.14)
trace("x created")
var y by delegate("Hello")
trace("y created")
var z by DelegateProvider(true)
trace("z created")
x eq 3.14
x = 1.62
x eq 1.62
y eq "Hello"
y = "Goodbye"
y eq "Goodbye"
z eq true
z = false
z eq false
trace eq """
creating DelegateProvider(3.14)
providing Delegate(3.14)
x created
creating DelegateProvider(Hello)
providing Delegate(Hello)
y created
providing Delegate(true)
z created
getValue: 3.14
setValue: 1.62
getValue: 1.62
getValue: Hello
setValue: Goodbye
getValue: Goodbye
getValue: true
setValue: false
getValue: false
"""
}

View File

@ -0,0 +1,16 @@
// DelegationTools/MapAccessors.kt
package delegationtools
import kotlin.reflect.KProperty
operator fun MutableMap<String, Any>.getValue(
thisRef: Any?, property: KProperty<*>
): Any? {
return this[property.name]
}
operator fun MutableMap<String, Any>.setValue(
thisRef: Any?, property: KProperty<*>,
value: Any
) {
this[property.name] = value
}

View File

@ -6,5 +6,7 @@ files:
visible: true
- name: src/NeverNull.kt
visible: true
- name: src/DelegateProvider.kt
- name: src/MapAccessors.kt
visible: true
- name: src/CarService.kt
visible: true

View File

@ -1,5 +1,5 @@
// PropertyDelegation/PropDelegationSoln1.kt
package propertyDelegationExercise1
package propertyDelegationToolsExercise1
import atomictest.eq
class Configuration(

View File

@ -3,10 +3,10 @@ files:
- name: src/Task.kt
visible: true
placeholders:
- offset: 103
- offset: 108
length: 141
placeholder_text: class Configuration
- offset: 259
- offset: 264
length: 323
placeholder_text: |-
/*

View File

@ -1,4 +1,4 @@
## Property Delegation (#1)
## Property Delegation Tools (#1)
Convert `Configuration.kt` to use a `Map` instead of a file. The starter code
in `main()` tests your new `Configuration` class.

View File

@ -1,9 +1,9 @@
package propertyDelegationExercise1
package propertyDelegationToolsExercise1
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationExercise1 {
class TestPropertyDelegationToolsExercise1 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -1,5 +1,5 @@
// DelegationTools/DelegToolsSoln1.kt
package propertyDelegationToolsExercise1
package propertyDelegationToolsExercise2
import kotlin.properties.Delegates
import atomictest.*

View File

@ -1,4 +1,4 @@
## Property Delegation Tools (#1)
## Property Delegation Tools (#2)
The starter code defines a `data class` called `Flag` containing a `Boolean`
called `b` with a default value of `false`. In `main()`, create three `var`s:

View File

@ -1,9 +1,9 @@
package propertyDelegationToolsExercise1
package propertyDelegationToolsExercise2
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationToolsExercise1 {
class TestPropertyDelegationToolsExercise2 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -1,5 +1,5 @@
// DelegationTools/DelegToolsSoln2.kt
package propertyDelegationToolsExercise2
package propertyDelegationToolsExercise3
import kotlin.properties.Delegates.observable
import kotlin.reflect.KProperty
import atomictest.*

View File

@ -1,4 +1,4 @@
## Property Delegation Tools (#2)
## Property Delegation Tools (#3)
Create a generic function `observe()` that uses `trace()` to capture
`"$propertyName $oldValue to $newValue"` whenever a property changes. Ensure

View File

@ -1,9 +1,9 @@
package propertyDelegationToolsExercise2
package propertyDelegationToolsExercise3
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationToolsExercise2 {
class TestPropertyDelegationToolsExercise3 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -1,5 +1,5 @@
// DelegationTools/DelegToolsSoln3.kt
package propertyDelegationToolsExercise3
package propertyDelegationToolsExercise4
import atomictest.eq
import kotlin.properties.Delegates

View File

@ -1,4 +1,4 @@
## Property Delegation Tools (#3)
## Property Delegation Tools (#4)
Create a `class PositiveInt(i: Int)` containing a property `var n` such that
any value assigned to `n` can only be greater than zero. The starter code in

View File

@ -1,9 +1,9 @@
package propertyDelegationToolsExercise3
package propertyDelegationToolsExercise4
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationToolsExercise3 {
class TestPropertyDelegationToolsExercise4 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -18,5 +18,3 @@ files:
visible: true
- name: src/Configuration.kt
visible: true
- name: src/CarService.kt
visible: true

View File

@ -1,5 +1,5 @@
// PropertyDelegation/PropDelegationSoln2.kt
package propertyDelegationExercise2
package propertyDelegationExercise1
import atomictest.eq
import kotlin.reflect.KProperty

View File

@ -1,4 +1,4 @@
## Property Delegation (#2)
## Property Delegation (#1)
Starting with `Add.kt`, add a `setValue()` extension function to `Sum` that
assigns its `value` argument to `a` in `Add` (change `Add` to make this work).

View File

@ -1,9 +1,9 @@
package propertyDelegationExercise2
package propertyDelegationExercise1
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationExercise2 {
class TestPropertyDelegationExercise1 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -1,5 +1,5 @@
// PropertyDelegation/PropDelegationSoln3.kt
package propertyDelegationExercise3
package propertyDelegationExercise2
import atomictest.eq
import kotlin.reflect.KProperty

View File

@ -1,4 +1,4 @@
## Property Delegation (#3)
## Property Delegation (#2)
Start with:

View File

@ -1,9 +1,9 @@
package propertyDelegationExercise3
package propertyDelegationExercise2
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationExercise3 {
class TestPropertyDelegationExercise2 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -1,5 +1,5 @@
// PropertyDelegation/PropDelegationSoln4.kt
package propertyDelegationExercise4
package propertyDelegationExercise3
import atomictest.eq
import kotlin.reflect.KProperty

View File

@ -1,4 +1,4 @@
## Property Delegation (#4)
## Property Delegation (#3)
Create a generic class that begins:

View File

@ -1,9 +1,9 @@
package propertyDelegationExercise4
package propertyDelegationExercise3
import org.junit.Test
import util.unimplementedTest
class TestPropertyDelegationExercise4 {
class TestPropertyDelegationExercise3 {
@Test fun testSolution() {
unimplementedTest()
}

View File

@ -0,0 +1,13 @@
// ScopeFunctions/AndNullability.kt
package scopefunctions
import atomictest.eq
import kotlin.random.Random
fun gets(): String? =
if (Random.nextBoolean()) "str!" else null
fun main() {
gets()?.let {
it.removeSuffix("!") + it.length
}?.eq("str4")
}

View File

@ -3,8 +3,9 @@ package scopefunctions
import atomictest.*
data class Blob(val id: Int) : AutoCloseable {
fun show() { trace("Show $id")}
override fun close() = trace("Close $id")
override fun toString() = "Blob($id)"
fun show() { trace("$this")}
override fun close() = trace("Close $this")
}
fun main() {
@ -19,20 +20,20 @@ fun main() {
Blob(9).also { it.show() }.apply { close() }
Blob(10).apply { show() }.use { }
trace eq """
Show 1
Show 2
Show 3
Show 4
Show 5
Show 6
Close 6
Show 7
Close 7
Show 8
Close 8
Show 9
Close 9
Show 10
Close 10
Blob(1)
Blob(2)
Blob(3)
Blob(4)
Blob(5)
Blob(6)
Close Blob(6)
Blob(7)
Close Blob(7)
Blob(8)
Close Blob(8)
Blob(9)
Close Blob(9)
Blob(10)
Close Blob(10)
"""
}

View File

@ -1,64 +0,0 @@
// ScopeFunctions/Differences.kt
package scopefunctions
import atomictest.eq
data class Tag(var n: Int = 0) {
var s: String = ""
fun increment() = ++n
}
fun main() {
// let(): Access object with 'it'
// Returns last expression in lambda
val let1: Int = Tag(1).let {
it.s = "let: ${it.n}"
it.increment()
}
let1 eq 2
// let() with named lambda argument:
val let2: Int = Tag(2).let { tag ->
tag.s = "let: ${tag.n}"
tag.increment()
}
let2 eq 3
// run(): Access object with 'this'
// Returns last expression in lambda
val run1: Int = Tag(3).run {
s = "run: $n"
increment()
}
run1 eq 4
// with(): Access object with 'this'
// Returns last expression in lambda
val with1: Int = with(Tag(4)) {
s = "with: $n"
increment()
}
with1 eq 5
// apply(): Access object with 'this'
// Returns modified object
val apply1: Tag = Tag(5).apply {
s = "apply: $n"
increment()
}
apply1 eq "Tag(n=6)"
// also(): Access object with 'it'
// Returns modified object
val also1: Tag = Tag(6).also {
it.s = "also: ${it.n}"
it.increment()
}
also1 eq "Tag(n=7)"
// also() with named lambda argument:
val also2: Tag = Tag(7).also { tag ->
tag.s = "also: ${tag.n}"
tag.increment()
}
also2 eq "Tag(n=8)"
}

View File

@ -9,7 +9,7 @@ fun display(map: Map<String, Plumbus>) {
val pb1: Plumbus = map["main"]?.let {
it.id += 10
it
} ?: return // [1]
} ?: return
trace(pb1)
val pb2: Plumbus? = map["main"]?.run {

View File

@ -6,23 +6,23 @@ val functions = listOf(
fun(name: String?) {
name
?.takeUnless { it.isBlank() }
?.let { trace("Hi! I am $it") }
?.let { trace("$it in let") }
},
fun(name: String?) {
name
?.takeUnless { it.isBlank() }
?.run { trace("Hi! I am $this") }
?.run { trace("$this in run") }
},
fun(name: String?) {
name
?.takeUnless { it.isBlank() }
?.apply { trace("Hi! I am $this") }
?.apply { trace("$this in apply") }
},
fun(name: String?) {
name
?.takeUnless { it.isBlank() }
?.also { trace("Hi! I am $it") }
}
?.also { trace("$it in also") }
},
)
fun main() {
@ -30,9 +30,9 @@ fun main() {
functions.forEach { it(" ") }
functions.forEach { it("Yumyulack") }
trace eq """
Hi! I am Yumyulack
Hi! I am Yumyulack
Hi! I am Yumyulack
Hi! I am Yumyulack
Yumyulack in let
Yumyulack in run
Yumyulack in apply
Yumyulack in also
"""
}

View File

@ -0,0 +1,34 @@
// ScopeFunctions/Nesting.kt
package scopefunctions
import atomictest.eq
fun nesting(s: String, i: Int): String =
with(s) {
with(i) {
toString()
}
} +
s.let {
i.let {
it.toString()
}
} +
s.run {
i.run {
toString()
}
} +
s.apply {
i.apply {
toString()
}
} +
s.also {
i.also {
it.toString()
}
}
fun main() {
nesting("X", 7) eq "777XX"
}

View File

@ -11,14 +11,14 @@ fun whichGnome(gnome: Gnome?) {
}
fun main() {
whichGnome(null)
whichGnome(Gnome("Bob"))
whichGnome(null)
trace eq """
null
Bob
Gnome: Bob
Gnome: Bob
Gnome: Bob
Gnome: Bob
null
"""
}

View File

@ -1,7 +1,5 @@
type: theory
files:
- name: src/Differences.kt
visible: true
- name: src/Gnome.kt
visible: true
- name: src/NullGnome.kt
@ -12,3 +10,7 @@ files:
visible: true
- name: src/Blob.kt
visible: true
- name: src/AndNullability.kt
visible: true
- name: src/Nesting.kt
visible: true

View File

@ -0,0 +1,14 @@
// UsingOperators/ComparableRange.kt
package usingoperators
import atomictest.eq
class F(val i: Int): Comparable<F> {
override fun compareTo(other: F) =
i.compareTo(other.i)
}
fun main() {
val range = F(1)..F(7)
(F(3) in range) eq true
(F(9) in range) eq false
}

View File

@ -6,9 +6,9 @@ data class Contact(
val name: String,
val mobile: String
): Comparable<Contact> {
override
fun compareTo(other: Contact): Int =
name.compareTo(other.name)
override fun compareTo(
other: Contact
): Int = name.compareTo(other.name)
}
fun main() {

View File

@ -16,3 +16,5 @@ files:
visible: true
- name: src/DestructuringData.kt
visible: true
- name: src/ComparableRange.kt
visible: true

View File

@ -1,6 +1,6 @@
content:
- Scope Functions
- Extension Lambdas
- Scope Functions
- Creating Generics
- Operator Overloading
- Using Operators

View File

@ -19,7 +19,7 @@ fun getTrace(fileName: String): List<String> {
fun main() {
DataFile("file_empty.txt").writeText("")
DataFile("file_real.txt").writeText(
DataFile("file_wubba.txt").writeText(
"wubba lubba dub dub")
capture {
getTrace("wrong_name.txt")
@ -33,6 +33,6 @@ fun main() {
getTrace("file_empty.txt")
} eq "IllegalArgumentException: " +
"file_empty.txt is empty"
getTrace("file_real.txt") eq
getTrace("file_wubba.txt") eq
"[wubba lubba dub dub]"
}

View File

@ -9,6 +9,6 @@ fun main() {
}
/* Output:
1. [OK] Everything is fine.
2. [Fail] Something has broken.
1 [OK] Everything is fine.
2 [Fail] Something has broken.
*/

View File

@ -1,19 +1,15 @@
// NullableTypes/Amphibian.kt
package nullabletypes
interface Amphibian
class Frog : Amphibian
class Amphibian
enum class Species {
Frog, Toad, Salamander, Caecilian
}
fun main() {
val a1: Amphibian = Frog()
val a1: Amphibian = Amphibian()
val a2: Amphibian? = null
val f1: Frog = Frog()
val f2: Frog? = null
val at1: Species = Species.Toad
val at2: Species? = null
}

View File

@ -38,7 +38,10 @@ fun List<Int>.myJoinToString(prefix: String, postfix: String): String {
fun main() {
val list = listOf(1, 2, 3)
list.myJoinToString() eq "1, 2, 3"
list.myJoinToString(separator = "|") eq "1|2|3"
list.myJoinToString(separator = "..", prefix = "Numbers: ") eq "Numbers: 1..2..3"
list.myJoinToString() eq
"MY: 1, 2, 3"
list.myJoinToString(separator = "|") eq
"MY: 1|2|3"
list.myJoinToString(separator = "..", prefix = "Numbers: ") eq
"MY: Numbers: 1..2..3"
}

View File

@ -2201,31 +2201,6 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Preventing Failure/Unit Testing/Examples/src/UsingExpect.kt", unittesting.UsingExpectKt::main);
}
@Test
public void testBlob() {
testExample("Power Tools/Scope Functions/Examples/src/Blob.kt", scopefunctions.BlobKt::main);
}
@Test
public void testNullGnome() {
testExample("Power Tools/Scope Functions/Examples/src/NullGnome.kt", scopefunctions.NullGnomeKt::main);
}
@Test
public void testMapLookup() {
testExample("Power Tools/Scope Functions/Examples/src/MapLookup.kt", scopefunctions.MapLookupKt::main);
}
@Test
public void testNameTag() {
testExample("Power Tools/Scope Functions/Examples/src/NameTag.kt", scopefunctions.NameTagKt::main);
}
@Test
public void testDifferences() {
testExample("Power Tools/Scope Functions/Examples/src/Differences.kt", scopefunctions.DifferencesKt::main);
}
@Test
public void testTransform() {
testExample("Power Tools/Extension Lambdas/Examples/src/Transform.kt", extensionlambdas.TransformKt::main);
@ -2271,6 +2246,36 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Power Tools/Extension Lambdas/Examples/src/ListsAndMaps.kt", extensionlambdas.ListsAndMapsKt::main);
}
@Test
public void testAndNullability() {
testExample("Power Tools/Scope Functions/Examples/src/AndNullability.kt", scopefunctions.AndNullabilityKt::main);
}
@Test
public void testBlob() {
testExample("Power Tools/Scope Functions/Examples/src/Blob.kt", scopefunctions.BlobKt::main);
}
@Test
public void testNullGnome() {
testExample("Power Tools/Scope Functions/Examples/src/NullGnome.kt", scopefunctions.NullGnomeKt::main);
}
@Test
public void testNesting() {
testExample("Power Tools/Scope Functions/Examples/src/Nesting.kt", scopefunctions.NestingKt::main);
}
@Test
public void testMapLookup() {
testExample("Power Tools/Scope Functions/Examples/src/MapLookup.kt", scopefunctions.MapLookupKt::main);
}
@Test
public void testNameTag() {
testExample("Power Tools/Scope Functions/Examples/src/NameTag.kt", scopefunctions.NameTagKt::main);
}
@Test
public void testSpeakers() {
testExample("Power Tools/Creating Generics/Examples/src/Speakers.kt", creatinggenerics.SpeakersKt::main);
@ -2342,8 +2347,23 @@ public class TestAllExamples extends AbstractTestExamples {
}
@Test
public void testAllOperators() {
testExample("Power Tools/Operator Overloading/Examples/src/AllOperators.kt", operatoroverloading.AllOperatorsKt::main);
public void testDefiningEquality() {
testExample("Power Tools/Operator Overloading/Examples/src/DefiningEquality.kt", operatoroverloading.DefiningEqualityKt::main);
}
@Test
public void testComparison() {
testExample("Power Tools/Operator Overloading/Examples/src/Comparison.kt", operatoroverloading.ComparisonKt::main);
}
@Test
public void testContainerAccess() {
testExample("Power Tools/Operator Overloading/Examples/src/ContainerAccess.kt", operatoroverloading.ContainerAccessKt::main);
}
@Test
public void testRanges() {
testExample("Power Tools/Operator Overloading/Examples/src/Ranges.kt", operatoroverloading.RangesKt::main);
}
@Test
@ -2351,16 +2371,41 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Power Tools/Operator Overloading/Examples/src/Invoke.kt", operatoroverloading.InvokeKt::main);
}
@Test
public void testConfusingPrecedence() {
testExample("Power Tools/Operator Overloading/Examples/src/ConfusingPrecedence.kt", operatoroverloading.ConfusingPrecedenceKt::main);
}
@Test
public void testMolecule() {
testExample("Power Tools/Operator Overloading/Examples/src/Molecule.kt", operatoroverloading.MoleculeKt::main);
}
@Test
public void testDifferentTypes() {
testExample("Power Tools/Operator Overloading/Examples/src/DifferentTypes.kt", operatoroverloading.DifferentTypesKt::main);
}
@Test
public void testDefaultEquality() {
testExample("Power Tools/Operator Overloading/Examples/src/DefaultEquality.kt", operatoroverloading.DefaultEqualityKt::main);
}
@Test
public void testEqualsForNullable() {
testExample("Power Tools/Operator Overloading/Examples/src/EqualsForNullable.kt", operatoroverloading.EqualsForNullableKt::main);
}
@Test
public void testNum() {
testExample("Power Tools/Operator Overloading/Examples/src/Num.kt", operatoroverloading.NumKt::main);
}
@Test
public void testArithmeticOperators() {
testExample("Power Tools/Operator Overloading/Examples/src/ArithmeticOperators.kt", operatoroverloading.ArithmeticOperatorsKt::main);
}
@Test
public void testCompareTo() {
testExample("Power Tools/Using Operators/Examples/src/CompareTo.kt", usingoperators.CompareToKt::main);
@ -2396,6 +2441,11 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Power Tools/Using Operators/Examples/src/DestructuringData.kt", usingoperators.DestructuringDataKt::main);
}
@Test
public void testComparableRange() {
testExample("Power Tools/Using Operators/Examples/src/ComparableRange.kt", usingoperators.ComparableRangeKt::main);
}
@Test
public void testDestructuringDuo() {
testExample("Power Tools/Using Operators/Examples/src/DestructuringDuo.kt", usingoperators.DestructuringDuoKt::main);
@ -2426,11 +2476,6 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Power Tools/Property Delegation/Examples/src/Add.kt", propertydelegation2.AddKt::main);
}
@Test
public void testCarService() {
testExample("Power Tools/Property Delegation/Examples/src/CarService.kt", propertydelegation.CarServiceKt::main);
}
@Test
public void testFibonacciProperty() {
testExample("Power Tools/Property Delegation/Examples/src/FibonacciProperty.kt", propertydelegation.FibonacciPropertyKt::main);
@ -2447,8 +2492,8 @@ public class TestAllExamples extends AbstractTestExamples {
}
@Test
public void testDelegateProvider() {
testExample("Power Tools/Property Delegation Tools/Examples/src/DelegateProvider.kt", delegationtools.DelegateProviderKt::main);
public void testCarService() {
testExample("Power Tools/Property Delegation Tools/Examples/src/CarService.kt", propertydelegation.CarServiceKt::main);
}
@Test
@ -2531,6 +2576,11 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Appendices/Java Interoperability/Examples/src/Interoperability/MakeSalad.java", Interoperability.MakeSalad::main);
}
@Test
public void testJavaWrapper() {
testExample("Appendices/Java Interoperability/Examples/src/Interoperability/JavaWrapper.java", Interoperability.JavaWrapper::main);
}
@Test
public void testCallTopLevelFunction() {
testExample("Appendices/Java Interoperability/Examples/src/Interoperability/CallTopLevelFunction.java", Interoperability.CallTopLevelFunction::main);
@ -2552,8 +2602,8 @@ public class TestAllExamples extends AbstractTestExamples {
}
@Test
public void testUsingDataClass() {
testExample("Appendices/Java Interoperability/Examples/src/Interoperability/UsingDataClass.java", Interoperability.UsingDataClass::main);
public void testUseDataClass() {
testExample("Appendices/Java Interoperability/Examples/src/Interoperability/UseDataClass.java", Interoperability.UseDataClass::main);
}
@Test
@ -2591,6 +2641,11 @@ public class TestAllExamples extends AbstractTestExamples {
testExample("Appendices/Java Interoperability/Examples/src/NPEOnPlatformType.kt", NPEOnPlatformTypeKt::main);
}
@Test
public void testKotlinWrapper() {
testExample("Appendices/Java Interoperability/Examples/src/KotlinWrapper.kt", interop.KotlinWrapperKt::main);
}
@Test
public void testTask() {
testExample("Programming Basics/Hello, World/Exercise 2/src/Task.kt", helloWorldExercise2.TaskKt::main);
@ -3943,34 +3998,34 @@ public class TestAllExamples extends AbstractTestExamples {
@Test
public void testTask265() {
testExample("Power Tools/Scope Functions/Exercise 1/src/Task.kt", scopeFunctionsExercise1.TaskKt::main);
}
@Test
public void testTask266() {
testExample("Power Tools/Scope Functions/Exercise 2/src/Task.kt", scopeFunctionsExercise2.TaskKt::main);
}
@Test
public void testTask267() {
testExample("Power Tools/Scope Functions/Exercise 3/src/Task.kt", scopeFunctionsExercise3.TaskKt::main);
}
@Test
public void testTask268() {
testExample("Power Tools/Extension Lambdas/Exercise 1/src/Task.kt", extensionLambdasExercise1.TaskKt::main);
}
@Test
public void testTask269() {
public void testTask266() {
testExample("Power Tools/Extension Lambdas/Exercise 2/src/Task.kt", extensionLambdasExercise2.TaskKt::main);
}
@Test
public void testTask270() {
public void testTask267() {
testExample("Power Tools/Extension Lambdas/Exercise 3/src/Task.kt", extensionLambdasExercise3.TaskKt::main);
}
@Test
public void testTask268() {
testExample("Power Tools/Scope Functions/Exercise 1/src/Task.kt", scopeFunctionsExercise1.TaskKt::main);
}
@Test
public void testTask269() {
testExample("Power Tools/Scope Functions/Exercise 2/src/Task.kt", scopeFunctionsExercise2.TaskKt::main);
}
@Test
public void testTask270() {
testExample("Power Tools/Scope Functions/Exercise 3/src/Task.kt", scopeFunctionsExercise3.TaskKt::main);
}
@Test
public void testTask271() {
testExample("Power Tools/Creating Generics/Exercise 1/src/Task.kt", creatingGenericsExercise1.TaskKt::main);
@ -4033,24 +4088,24 @@ public class TestAllExamples extends AbstractTestExamples {
@Test
public void testTask283() {
testExample("Power Tools/Property Delegation/Exercise 4/src/Task.kt", propertyDelegationExercise4.TaskKt::main);
}
@Test
public void testTask284() {
testExample("Power Tools/Property Delegation Tools/Exercise 1/src/Task.kt", propertyDelegationToolsExercise1.TaskKt::main);
}
@Test
public void testTask285() {
public void testTask284() {
testExample("Power Tools/Property Delegation Tools/Exercise 2/src/Task.kt", propertyDelegationToolsExercise2.TaskKt::main);
}
@Test
public void testTask286() {
public void testTask285() {
testExample("Power Tools/Property Delegation Tools/Exercise 3/src/Task.kt", propertyDelegationToolsExercise3.TaskKt::main);
}
@Test
public void testTask286() {
testExample("Power Tools/Property Delegation Tools/Exercise 4/src/Task.kt", propertyDelegationToolsExercise4.TaskKt::main);
}
@Test
public void testTask287() {
testExample("Power Tools/Lazy Initialization/Exercise 1/src/Task.kt", lazyInitializationExercise1.TaskKt::main);

View File

@ -777,22 +777,22 @@ class TestAllExercises : AbstractTestExercises() {
fun `test#256 UnitTesting Exercise#3`() = testClass(unitTestingExercise3.TestUnitTestingExercise3::class)
@Test
fun `test#257 ScopeFunctions Exercise#1`() = testClass(scopeFunctionsExercise1.TestScopeFunctionsExercise1::class)
fun `test#257 ExtensionLambdas Exercise#1`() = testClass(extensionLambdasExercise1.TestExtensionLambdasExercise1::class)
@Test
fun `test#258 ScopeFunctions Exercise#2`() = testClass(scopeFunctionsExercise2.TestScopeFunctionsExercise2::class)
fun `test#258 ExtensionLambdas Exercise#2`() = testClass(extensionLambdasExercise2.TestExtensionLambdasExercise2::class)
@Test
fun `test#259 ScopeFunctions Exercise#3`() = testClass(scopeFunctionsExercise3.TestScopeFunctionsExercise3::class)
fun `test#259 ExtensionLambdas Exercise#3`() = testClass(extensionLambdasExercise3.TestExtensionLambdasExercise3::class)
@Test
fun `test#260 ExtensionLambdas Exercise#1`() = testClass(extensionLambdasExercise1.TestExtensionLambdasExercise1::class)
fun `test#260 ScopeFunctions Exercise#1`() = testClass(scopeFunctionsExercise1.TestScopeFunctionsExercise1::class)
@Test
fun `test#261 ExtensionLambdas Exercise#2`() = testClass(extensionLambdasExercise2.TestExtensionLambdasExercise2::class)
fun `test#261 ScopeFunctions Exercise#2`() = testClass(scopeFunctionsExercise2.TestScopeFunctionsExercise2::class)
@Test
fun `test#262 ExtensionLambdas Exercise#3`() = testClass(extensionLambdasExercise3.TestExtensionLambdasExercise3::class)
fun `test#262 ScopeFunctions Exercise#3`() = testClass(scopeFunctionsExercise3.TestScopeFunctionsExercise3::class)
@Test
fun `test#263 CreatingGenerics Exercise#1`() = testClass(creatingGenericsExercise1.TestCreatingGenericsExercise1::class)
@ -831,16 +831,16 @@ class TestAllExercises : AbstractTestExercises() {
fun `test#274 PropertyDelegation Exercise#3`() = testClass(propertyDelegationExercise3.TestPropertyDelegationExercise3::class)
@Test
fun `test#275 PropertyDelegation Exercise#4`() = testClass(propertyDelegationExercise4.TestPropertyDelegationExercise4::class)
fun `test#275 PropertyDelegationTools Exercise#1`() = testClass(propertyDelegationToolsExercise1.TestPropertyDelegationToolsExercise1::class)
@Test
fun `test#276 PropertyDelegationTools Exercise#1`() = testClass(propertyDelegationToolsExercise1.TestPropertyDelegationToolsExercise1::class)
fun `test#276 PropertyDelegationTools Exercise#2`() = testClass(propertyDelegationToolsExercise2.TestPropertyDelegationToolsExercise2::class)
@Test
fun `test#277 PropertyDelegationTools Exercise#2`() = testClass(propertyDelegationToolsExercise2.TestPropertyDelegationToolsExercise2::class)
fun `test#277 PropertyDelegationTools Exercise#3`() = testClass(propertyDelegationToolsExercise3.TestPropertyDelegationToolsExercise3::class)
@Test
fun `test#278 PropertyDelegationTools Exercise#3`() = testClass(propertyDelegationToolsExercise3.TestPropertyDelegationToolsExercise3::class)
fun `test#278 PropertyDelegationTools Exercise#4`() = testClass(propertyDelegationToolsExercise4.TestPropertyDelegationToolsExercise4::class)
@Test
fun `test#279 LazyInitialization Exercise#1`() = testClass(lazyInitializationExercise1.TestLazyInitializationExercise1::class)