Files
kotlin/libraries/stdlib/test/numbers/FloorDivModTest.kt
2021-08-10 17:57:50 +02:00

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)
}
}
}