org.firebirdsql.decimal.Decimal128 Maven / Gradle / Ivy
Show all versions of decimal-java Show documentation
/*
* Copyright (c) 2018 Firebird development team and individual contributors
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
package org.firebirdsql.decimal;
import java.math.BigDecimal;
import java.math.BigInteger;
/**
* An IEEE-754 Decimal128.
*
* @author Mark Rotteveel
*/
public final class Decimal128 extends Decimal {
public static final Decimal128 POSITIVE_INFINITY = new Decimal128(Signum.POSITIVE, DecimalType.INFINITY);
public static final Decimal128 NEGATIVE_INFINITY = new Decimal128(Signum.NEGATIVE, DecimalType.INFINITY);
public static final Decimal128 POSITIVE_NAN = new Decimal128(Signum.POSITIVE, DecimalType.NAN);
public static final Decimal128 NEGATIVE_NAN = new Decimal128(Signum.NEGATIVE, DecimalType.NAN);
public static final Decimal128 POSITIVE_SIGNALING_NAN = new Decimal128(Signum.POSITIVE, DecimalType.SIGNALING_NAN);
public static final Decimal128 NEGATIVE_SIGNALING_NAN = new Decimal128(Signum.NEGATIVE, DecimalType.SIGNALING_NAN);
private static final Decimal128Factory DECIMAL_128_FACTORY = new Decimal128Factory();
private static final DecimalCodec DECIMAL_128_CODEC = new DecimalCodec<>(DECIMAL_128_FACTORY);
private Decimal128(int signum, DecimalType decimalType) {
super(signum, decimalType);
}
private Decimal128(int signum, BigDecimal bigDecimal) {
super(signum, bigDecimal);
}
@Override
DecimalCodec getDecimalCodec() {
return DECIMAL_128_CODEC;
}
@Override
DecimalFactory getDecimalFactory() {
return DECIMAL_128_FACTORY;
}
/**
* Parses the provided byte array to a {@code Decimal128}.
*
* This method parses network byte-order (aka big-endian). When using little-endian order, you will need to
* reverse the bytes in the array first.
*
*
* @param decBytes
* Bytes of the Decimal128 value in network byte-order (aka big-endian)
* @return Instance of {@code Decimal128}
* @throws IllegalArgumentException
* When {@code decBytes} is not 16 bytes long
*/
public static Decimal128 parseBytes(final byte[] decBytes) {
return DECIMAL_128_CODEC.parseBytes(decBytes);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* Values exceeding the range of this type will be returned as +/-Infinity.
*
*
* @param value
* Big decimal value to convert
* @return Decimal128 equivalent
*/
public static Decimal128 valueOf(final BigDecimal value) {
return valueOf(value, OverflowHandling.ROUND_TO_INFINITY);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* Values exceeding the range of this type will be handled according to the specified overflow handling.
*
*
* @param value
* Big decimal value to convert
* @param overflowHandling
* Overflow handling to apply
* @return Decimal128 equivalent
* @throws DecimalOverflowException
* If {@code OverflowHandling#THROW_EXCEPTION} and the value is out of range.
*/
public static Decimal128 valueOf(final BigDecimal value, final OverflowHandling overflowHandling) {
return DECIMAL_128_FACTORY.valueOf(value, overflowHandling);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* Values exceeding the range of this type will be returned as +/-Infinity.
*
*
* @param value
* Big integer value to convert
* @return Decimal128 equivalent
*/
public static Decimal128 valueOf(final BigInteger value) {
return valueOf(value, OverflowHandling.ROUND_TO_INFINITY);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* Values exceeding the range of this type will be handled according to the specified overflow handling.
*
*
* Calling this method is equivalent to {@code valueOf(new BigDecimal(value), overflowHandling)}.
*
*
* @param value
* Big integer value to convert
* @param overflowHandling
* Handling of overflows
* @return Decimal128 equivalent
* @throws DecimalOverflowException
* If {@code OverflowHandling#THROW_EXCEPTION} and the value is out of range.
* @see #valueOfExact(BigInteger)
*/
public static Decimal128 valueOf(final BigInteger value, final OverflowHandling overflowHandling) {
return DECIMAL_128_FACTORY.valueOf(value, overflowHandling);
}
/**
* Creates a {@code Decimal128} from {@code value}, rejecting values that would lose precision due to rounding.
*
* @param value Big integer value to convert
* @throws DecimalOverflowException
* If the value is out of range.
* @return Decimal128 equivalent
* @see #valueOf(BigInteger, OverflowHandling)
*/
public static Decimal128 valueOfExact(final BigInteger value) {
return DECIMAL_128_FACTORY.valueOfExact(value);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* {@code Double.NaN} is mapped to positive NaN, the infinities to their equivalent +/- infinity.
*
*
* For normal, finite, values, this is equivalent to {@code valueOf(BigDecimal.valueOf(value))}.
*
*
* @param value
* Double value
* @return Decimal equivalent
*/
public static Decimal128 valueOf(final double value) {
return valueOf(value, OverflowHandling.ROUND_TO_INFINITY);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* {@code Double.NaN} is mapped to positive NaN, the infinities to their equivalent +/- infinity.
*
*
* For normal, finite, values, this is equivalent to {@code valueOf(BigDecimal.valueOf(value), overflowHandling)}.
*
*
* @param value
* Double value
* @param overflowHandling
* Overflow handling to apply
* @return Decimal equivalent
* @throws DecimalOverflowException
* If {@code OverflowHandling#THROW_EXCEPTION} and the value is out of range.
*/
public static Decimal128 valueOf(final double value, final OverflowHandling overflowHandling) {
return DECIMAL_128_FACTORY.valueOf(value, overflowHandling);
}
/**
* Converts a decimal to Decimal128.
*
* For normal, finite, decimals, this behaves like {@code valueOf(decimal.toBigDecimal())}, see
* {@link #valueOf(BigDecimal)}.
*
*
* @param decimal
* Decimal to convert
* @return Decimal converted to Decimal128, or {@code decimal} itself if it already is Decimal128
*/
public static Decimal128 valueOf(Decimal> decimal) {
return valueOf(decimal, OverflowHandling.ROUND_TO_INFINITY);
}
/**
* Converts a decimal to Decimal128.
*
* For normal, finite, decimals, this behaves like {@code valueOf(decimal.toBigDecimal(), overflowHandling)}, see
* {@link #valueOf(BigDecimal, OverflowHandling)}.
*
*
* @param decimal
* Decimal to convert
* @param overflowHandling
* Overflow handling to apply
* @return Decimal converted to Decimal128, or {@code decimal} itself if it already is Decimal128
* @throws DecimalOverflowException
* If {@code OverflowHandling#THROW_EXCEPTION} and the value is out of range.
*/
public static Decimal128 valueOf(Decimal> decimal, OverflowHandling overflowHandling) {
if (decimal instanceof Decimal128) {
return (Decimal128) decimal;
}
return DECIMAL_128_FACTORY.valueOf(decimal, overflowHandling);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* Except for the special values [+/-]Inf, [+/-]Infinity, [+/-]NaN and [+/-]sNaN (case insensitive), the rules
* of {@link BigDecimal#BigDecimal(String)} apply, with special handling in place to discern between positive
* and negative zero.
*
*
* Values exceeding the range of this type will be returned as +/-Infinity.
*
*
* @param value
* String value to convert
* @return Decimal equivalent
* @throws NumberFormatException
* If the provided string is not valid numeric string.
*/
public static Decimal128 valueOf(final String value) {
return valueOf(value, OverflowHandling.ROUND_TO_INFINITY);
}
/**
* Creates a {@code Decimal128} from {@code value}, applying rounding where necessary.
*
* Except for the special values [+/-]Inf, [+/-]Infinity, [+/-]NaN and [+/-]sNaN (case insensitive), the rules
* of {@link BigDecimal#BigDecimal(String)} apply, with special handling in place to discern between positive
* and negative zero.
*
*
* Values exceeding the range of this type will be handled according to the specified overflow handling.
*
*
* @param value
* String value to convert
* @param overflowHandling
* Overflow handling to apply
* @return Decimal equivalent
* @throws NumberFormatException
* If the provided string is not valid numeric string.
* @throws DecimalOverflowException
* If {@code OverflowHandling#THROW_EXCEPTION} and the value is out of range.
*/
public static Decimal128 valueOf(final String value, final OverflowHandling overflowHandling) {
return DECIMAL_128_FACTORY.valueOf(value, overflowHandling);
}
private static class Decimal128Factory extends AbstractDecimalFactory {
private Decimal128Factory() {
super(Decimal128.class, DecimalFormat.Decimal128,
POSITIVE_INFINITY, NEGATIVE_INFINITY,
POSITIVE_NAN, NEGATIVE_NAN,
POSITIVE_SIGNALING_NAN, NEGATIVE_SIGNALING_NAN);
}
@Override
public Decimal128 createDecimal(int signum, BigDecimal value) {
return new Decimal128(signum, validateRange(value));
}
}
}