mirror of
https://github.com/jlengrand/kotlin.git
synced 2026-03-10 08:31:29 +00:00
285 lines
9.8 KiB
Kotlin
285 lines
9.8 KiB
Kotlin
/*
|
|
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
|
|
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
|
|
*/
|
|
|
|
package test.numbers
|
|
|
|
import kotlin.math.pow
|
|
import kotlin.math.sign
|
|
import kotlin.random.Random
|
|
import kotlin.test.*
|
|
|
|
class FloorDivModTest {
|
|
|
|
@Test
|
|
fun intDivMod() {
|
|
fun check(a: Int, b: Int, expectedFd: Int? = null, expectedMod: Int? = null) {
|
|
val div = a / b
|
|
val rem = a % b
|
|
val fd = a.floorDiv(b)
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedFd?.let { assertEquals(it, fd) }
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(div - if (a.sign != b.sign && rem != 0) 1 else 0, fd)
|
|
assertEquals(a - b * fd, mod)
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, div: $div, rem: $rem, floorDiv: $fd, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(10, -3, -4, -2)
|
|
check(10, 3, 3, 1)
|
|
check(-10, 3, -4, 2)
|
|
check(-10, -3, 3, -1)
|
|
val values = listOf(1, -1, 2, -2, 3, -3, Int.MIN_VALUE, Int.MAX_VALUE)
|
|
for (a in values + 0) {
|
|
for (b in values) {
|
|
check(a, b)
|
|
}
|
|
}
|
|
repeat(1000) {
|
|
val a = Random.nextInt()
|
|
val b = Random.nextInt().let { if (it == 0) 1 else it }
|
|
check(a, b)
|
|
}
|
|
}
|
|
|
|
@Test
|
|
fun longDivMod() {
|
|
fun check(a: Long, b: Long, expectedFd: Long? = null, expectedMod: Long? = null) {
|
|
val div = a / b
|
|
val rem = a % b
|
|
val fd = a.floorDiv(b)
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedFd?.let { assertEquals(it, fd) }
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(div - if (a.sign != b.sign && rem != 0L) 1 else 0, fd)
|
|
assertEquals(a - b * fd, mod)
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, div: $div, rem: $rem, floorDiv: $fd, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(10, -3, -4, -2)
|
|
check(10, 3, 3, 1)
|
|
check(-10, 3, -4, 2)
|
|
check(-10, -3, 3, -1)
|
|
val values = listOf(1, -1, 2, -2, 3, -3, Long.MIN_VALUE, Long.MAX_VALUE)
|
|
for (a in values + 0) {
|
|
for (b in values) {
|
|
check(a, b)
|
|
}
|
|
}
|
|
repeat(1000) {
|
|
val a = Random.nextLong()
|
|
val b = Random.nextLong().let { if (it == 0L) 1 else it }
|
|
check(a, b)
|
|
}
|
|
}
|
|
|
|
@Test
|
|
fun byteDivMod() {
|
|
fun check(a: Byte, b: Byte, expectedFd: Int? = null, expectedMod: Byte? = null) {
|
|
val div = a / b
|
|
val rem = a % b
|
|
val fd = a.floorDiv(b)
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedFd?.let { assertEquals(it, fd) }
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(div - if (a.toInt().sign != b.toInt().sign && rem != 0) 1 else 0, fd)
|
|
assertEquals(a - b * fd, mod.toInt())
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, div: $div, rem: $rem, floorDiv: $fd, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(10, -3, -4, -2)
|
|
check(10, 3, 3, 1)
|
|
check(-10, 3, -4, 2)
|
|
check(-10, -3, 3, -1)
|
|
val values = listOf(1, -1, 2, -2, 3, -3, Byte.MIN_VALUE, Byte.MAX_VALUE)
|
|
for (a in values + 0) {
|
|
for (b in values) {
|
|
check(a, b)
|
|
}
|
|
}
|
|
repeat(1000) {
|
|
val a = Random.nextInt().toByte()
|
|
val b = Random.nextInt().toByte().let { if (it == 0.toByte()) 1 else it }
|
|
check(a, b)
|
|
}
|
|
}
|
|
|
|
@Test
|
|
fun shortDivMod() {
|
|
fun check(a: Short, b: Short, expectedFd: Int? = null, expectedMod: Short? = null) {
|
|
val div = a / b
|
|
val rem = a % b
|
|
val fd = a.floorDiv(b)
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedFd?.let { assertEquals(it, fd) }
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(div - if (a.toInt().sign != b.toInt().sign && rem != 0) 1 else 0, fd)
|
|
assertEquals(a - b * fd, mod.toInt())
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, div: $div, rem: $rem, floorDiv: $fd, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(10, -3, -4, -2)
|
|
check(10, 3, 3, 1)
|
|
check(-10, 3, -4, 2)
|
|
check(-10, -3, 3, -1)
|
|
val values = listOf(1, -1, 2, -2, 3, -3, Short.MIN_VALUE, Short.MAX_VALUE)
|
|
for (a in values + 0) {
|
|
for (b in values) {
|
|
check(a, b)
|
|
}
|
|
}
|
|
repeat(1000) {
|
|
val a = Random.nextInt().toShort()
|
|
val b = Random.nextInt().toShort().let { if (it == 0.toShort()) 1 else it }
|
|
check(a, b)
|
|
}
|
|
}
|
|
|
|
@Test
|
|
fun longIntMod() {
|
|
fun check(a: Long, b: Int, expectedFd: Long? = null, expectedMod: Int? = null) {
|
|
val div = a / b
|
|
val rem = a % b
|
|
val fd = a.floorDiv(b)
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedFd?.let { assertEquals(it, fd) }
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(div - if (a.sign != b.sign && rem != 0L) 1 else 0, fd)
|
|
assertEquals(a - b * fd, mod.toLong())
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, div: $div, rem: $rem, floorDiv: $fd, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(Long.MAX_VALUE, 2, Long.MAX_VALUE / 2, 1)
|
|
@Suppress("INTEGER_OPERATOR_RESOLVE_WILL_CHANGE") // KT-47729
|
|
check(Long.MAX_VALUE, 1.shl(30), expectedMod = 1.shl(30) - 1)
|
|
@Suppress("INTEGER_OPERATOR_RESOLVE_WILL_CHANGE") // KT-47729
|
|
check(-1L, 1.shl(30), expectedMod = 1.shl(30) - 1)
|
|
check(Long.MAX_VALUE, Int.MAX_VALUE, expectedMod = 1)
|
|
check(Long.MAX_VALUE, Int.MIN_VALUE, expectedMod = -1)
|
|
}
|
|
|
|
@Test
|
|
fun shortIntMod() {
|
|
fun check(a: Short, b: Int, expectedFd: Int? = null, expectedMod: Int? = null) {
|
|
val div = a / b
|
|
val rem = a % b
|
|
val fd = a.floorDiv(b)
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedFd?.let { assertEquals(it, fd) }
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(div - if (a.toInt().sign != b.sign && rem != 0) 1 else 0, fd)
|
|
assertEquals(a.toInt() - b * fd, mod) // a.toInt() to workaround Int-out-of-range in JS legacy
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, div: $div, rem: $rem, floorDiv: $fd, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(Short.MAX_VALUE, Int.MAX_VALUE, 0, Short.MAX_VALUE.toInt())
|
|
check(Short.MAX_VALUE, Int.MIN_VALUE, -1, Int.MIN_VALUE + Short.MAX_VALUE)
|
|
@Suppress("INTEGER_OPERATOR_RESOLVE_WILL_CHANGE") // KT-47729
|
|
check((-1).toShort(), 1.shl(30), -1, 1.shl(30) - 1)
|
|
}
|
|
|
|
@Test
|
|
fun doubleMod() {
|
|
fun check(a: Double, b: Double, expectedMod: Double? = null) {
|
|
val rem = a % b
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(if (rem.sign == mod.sign) rem else rem + b, mod)
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, rem: $rem, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(10.125, -0.5, -0.375)
|
|
check(10.125, 0.5, 0.125)
|
|
check(-10.125, 0.5, 0.375)
|
|
check(-10.125, -0.5, -0.125)
|
|
val large = 2.0.pow(53)
|
|
check(0.025, large, 0.025)
|
|
check(-0.025, large, expectedMod = large)
|
|
check(0.025, -large, expectedMod = -large)
|
|
check(1.0, Double.NaN, Double.NaN)
|
|
check(Double.NaN, 1.0, Double.NaN)
|
|
check(Double.POSITIVE_INFINITY, Double.POSITIVE_INFINITY, Double.NaN)
|
|
val values = listOf(1.0, -1.0, 3.0, -3.0, large, -large, Double.MIN_VALUE, Double.MAX_VALUE,
|
|
Double.NEGATIVE_INFINITY, Double.POSITIVE_INFINITY, Double.NaN)
|
|
for (a in values + 0.0) {
|
|
for (b in values) {
|
|
check(a, b)
|
|
}
|
|
}
|
|
repeat(1000) {
|
|
val a = Random.nextDouble()
|
|
val b = Random.nextDouble().let { if (it == 0.0) 1.0 else it }
|
|
check(a, b)
|
|
}
|
|
}
|
|
@Test
|
|
fun floatMod() {
|
|
fun check(a: Float, b: Float, expectedMod: Float? = null) {
|
|
val rem = a % b
|
|
val mod = a.mod(b)
|
|
|
|
try {
|
|
expectedMod?.let { assertEquals(it, mod) }
|
|
assertEquals(if (rem.sign == mod.sign) rem else rem + b, mod)
|
|
} catch (e: AssertionError) {
|
|
fail("a: $a, b: $b, rem: $rem, mod: $mod", e)
|
|
}
|
|
}
|
|
|
|
check(10.125f, -0.5f, -0.375f)
|
|
check(10.125f, 0.5f, 0.125f)
|
|
check(-10.125f, 0.5f, 0.375f)
|
|
check(-10.125f, -0.5f, -0.125f)
|
|
val large = 2.0f.pow(53)
|
|
check(0.025f, large, 0.025f)
|
|
check(-0.025f, large, expectedMod = large)
|
|
check(0.025f, -large, expectedMod = -large)
|
|
check(1.0f, Float.NaN, Float.NaN)
|
|
check(Float.NaN, 1.0f, Float.NaN)
|
|
check(Float.POSITIVE_INFINITY, Float.POSITIVE_INFINITY, Float.NaN)
|
|
val values = listOf(1.0f, -1.0f, 3.0f, -3.0f, large, -large, Float.MIN_VALUE, Float.MAX_VALUE,
|
|
Float.NEGATIVE_INFINITY, Float.POSITIVE_INFINITY, Float.NaN)
|
|
for (a in values + 0.0f) {
|
|
for (b in values) {
|
|
check(a, b)
|
|
}
|
|
}
|
|
repeat(1000) {
|
|
val a = Random.nextFloat()
|
|
val b = Random.nextFloat().let { if (it == 0.0f) 1.0f else it }
|
|
check(a, b)
|
|
}
|
|
}
|
|
|
|
}
|
|
|