All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.firebirdsql.decimal.Decimal128 Maven / Gradle / Ivy

Go to download

A library to convert java.math.BigDecimal to and from IEEE-754r (IEEE-754-2008) decimal byte representations.

There is a newer version: 2.0.0
Show newest version
/*
 * 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)); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy