ToDoubleConversion.java
package org.klojang.convert;
import static org.klojang.convert.NumberMethods.MAX_DOUBLE_BD;
import static org.klojang.convert.NumberMethods.yes;
import static org.klojang.convert.TypeConversionException.inputTypeNotSupported;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Map;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
import java.util.function.Predicate;
final class ToDoubleConversion {
private ToDoubleConversion() {
throw new UnsupportedOperationException();
}
private static final Map<Class<?>, Predicate<Number>> fitsIntoDouble = Map.of(
BigDecimal.class, ToDoubleConversion::testBigDecimal,
BigInteger.class, ToDoubleConversion::testBigInteger,
Double.class, yes(),
Float.class, yes(),
Long.class, yes(),
AtomicLong.class, yes(),
Integer.class, yes(),
AtomicInteger.class, yes(),
Short.class, yes(),
Byte.class, yes()
);
static boolean isLossless(Number n) {
Predicate<Number> tester = fitsIntoDouble.get(n.getClass());
if (tester != null) {
return tester.test(n);
}
throw inputTypeNotSupported(n, Double.class);
}
static double exec(Number n) {
if (isLossless(n)) {
return n.doubleValue();
}
throw new TypeConversionException(n, Double.class);
}
private static boolean testBigInteger(Number n) {
return new BigDecimal(((BigInteger) n).abs())
.compareTo(MAX_DOUBLE_BD) <= 0;
}
private static boolean testBigDecimal(Number n) {
return ((BigDecimal) n).abs()
.compareTo(MAX_DOUBLE_BD) <= 0;
}
}