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

software.amazon.awssdk.core.SdkNumber Maven / Gradle / Ivy

/*
 * Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 * You may not use this file except in compliance with the License.
 * A copy of the License is located at
 *
 *  http://aws.amazon.com/apache2.0
 *
 * or in the "license" file accompanying this file. This file is distributed
 * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
 * express or implied. See the License for the specific language governing
 * permissions and limitations under the License.
 */

package software.amazon.awssdk.core;

import java.io.Serializable;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Objects;
import software.amazon.awssdk.annotations.Immutable;
import software.amazon.awssdk.annotations.SdkPublicApi;
import software.amazon.awssdk.utils.Validate;

/**
 * An in-memory representation of Number being given to a service or being returned by a service.
 * This is a SDK representation of a Number. This allows conversion to any desired numeric type by providing constructors
 * as below
 * 
    *
  • {@link #fromBigDecimal(BigDecimal)} to create from a BigDecimal.
  • *
  • {@link #fromBigInteger(BigInteger)} to create from a BigInteger.
  • *
  • {@link #fromDouble(double)} to create from a double
  • *
  • {@link #fromFloat(float)} to create from a float.
  • *
  • {@link #fromLong(long)} to create from a long.
  • *
  • {@link #fromShort(short)} to create from a short.
  • *
  • {@link #fromInteger(int)} to create from an integer.
  • *
  • {@link #fromString(String)} to create from a String
  • *
* * Thus, by doing this, this class is able to preserve arbitrary precision of any given number. *

* If {@link SdkNumber} is expected in a particular number format then its corresponding getter methods can be used. *

* Example for a {@link SdkNumber} created with {@link BigDecimal} the * {@link #fromBigDecimal(BigDecimal)} can be used. */ @SdkPublicApi @Immutable public final class SdkNumber extends Number implements Serializable { private static final long serialVersionUID = 1L; private final Number numberValue; private final String stringValue; /** * @param value Number value as passed in the from COnstructor. * @see #fromBigDecimal(BigDecimal) * @see #fromBigInteger(BigInteger) * @see #fromDouble(double) * @see #fromFloat(float) * @see #fromLong(long) * @see #fromShort(short) * @see #fromInteger(int) */ private SdkNumber(Number value) { this.numberValue = value; this.stringValue = null; } /** * . * * @param stringValue String value. * @see #fromString(String) */ private SdkNumber(String stringValue) { this.stringValue = stringValue; this.numberValue = null; } private static boolean isNumberValueNaN(Number numberValue) { return (numberValue instanceof Double && Double.isNaN((double) numberValue)) || (numberValue instanceof Float && Float.isNaN((float) numberValue)); } private static boolean isNumberValueInfinite(Number numberValue) { return (numberValue instanceof Double && Double.isInfinite((double) numberValue)) || (numberValue instanceof Float && Float.isInfinite((float) numberValue)); } private static Number valueOf(Number numberValue) { Number valueOfInfiniteOrNaN = valueOfInfiniteOrNaN(numberValue); return valueOfInfiniteOrNaN != null ? valueOfInfiniteOrNaN : valueInBigDecimal(numberValue); } private static Number valueOfInfiniteOrNaN(Number numberValue) { if (numberValue instanceof Double && (Double.isInfinite((double) numberValue) || Double.isNaN((double) numberValue))) { return numberValue.doubleValue(); } else if ((numberValue instanceof Float && (Float.isInfinite((float) numberValue) || Float.isNaN((float) numberValue)))) { return numberValue.floatValue(); } else { return null; } } /** * This function converts a given number to BigDecimal Number where the caller can convert to an primitive number. * This is done to keep the precision. * * @param numberValue The number value. * @return Big Decimal value for the given number. */ private static BigDecimal valueInBigDecimal(Number numberValue) { if (numberValue instanceof Double) { return BigDecimal.valueOf((double) numberValue); } else if (numberValue instanceof Float) { return BigDecimal.valueOf((float) numberValue); } else if (numberValue instanceof Integer) { return new BigDecimal((int) numberValue); } else if (numberValue instanceof Short) { return new BigDecimal((short) numberValue); } else if (numberValue instanceof Long) { return BigDecimal.valueOf((Long) numberValue); } else if (numberValue instanceof BigDecimal) { return (BigDecimal) numberValue; } else if (numberValue instanceof BigInteger) { return new BigDecimal((BigInteger) numberValue); } else { return new BigDecimal(numberValue.toString()); } } /** * Create {@link SdkNumber} from a integer value. * * @param integerValue Integer value. * @return new {@link SdkNumber} for the given int value. */ public static SdkNumber fromInteger(int integerValue) { return new SdkNumber(integerValue); } /** * Create {@link SdkNumber} from a BigInteger value. * * @param bigIntegerValue BigInteger value. * @return new {@link SdkNumber} for the given BigInteger value. */ public static SdkNumber fromBigInteger(BigInteger bigIntegerValue) { return new SdkNumber(bigIntegerValue); } /** * Create {@link SdkNumber} from a BigDecimal value. * * @param bigDecimalValue BigInteger value. * @return new {@link SdkNumber} for the given BigDecimal value. */ public static SdkNumber fromBigDecimal(BigDecimal bigDecimalValue) { Validate.notNull(bigDecimalValue, "BigDecimal cannot be null"); return new SdkNumber(bigDecimalValue); } /** * Create {@link SdkNumber} from a long Value. * * @param longValue long value. * @return new {@link SdkNumber} for the given long value. */ public static SdkNumber fromLong(long longValue) { return new SdkNumber(longValue); } /** * Create {@link SdkNumber} from a double Value. * * @param doubleValue long value. * @return new {@link SdkNumber} for the given double value. */ public static SdkNumber fromDouble(double doubleValue) { return new SdkNumber(doubleValue); } /** * Create {@link SdkNumber} from a long Value. * * @param shortValue long value. * @return new {@link SdkNumber} for the given long value. */ public static SdkNumber fromShort(short shortValue) { return new SdkNumber(shortValue); } /** * Create {@link SdkNumber} from a float Value. * * @param floatValue float value. * @return new {@link SdkNumber} for the given float value. */ public static SdkNumber fromFloat(float floatValue) { return new SdkNumber(floatValue); } /** * Create {@link SdkNumber} from a long Value. * * @param stringValue String value. * @return new {@link SdkNumber} for the given stringValue value. */ public static SdkNumber fromString(String stringValue) { return new SdkNumber(stringValue); } /** * Gets the integer value of the {@link SdkNumber}. * If we do a intValue() for {@link SdkNumber} constructed * from float, double, long, BigDecimal, BigInteger number type then it * may result in loss of magnitude and a loss of precision. * The result may lose some of the least significant bits of the value. * Precision is not lost while getting a {@link SdkNumber} which was constructed as * lower precision number type like short, byte, integer. * * @return integer value of {@link SdkNumber} . */ @Override public int intValue() { if (numberValue instanceof Integer) { return numberValue.intValue(); } if (stringValue != null) { return new BigDecimal(stringValue).intValue(); } return valueOf(numberValue).intValue(); } /** * Gets the long value of the {@link SdkNumber}. * If we do a longValue() for {@link SdkNumber} constructed from * float, double, BigDecimal, BigInteger number type then it * may result in loss of magnitude and a loss of precision. * Precision is not lost while getting a {@link SdkNumber} which was constructed from * lower precision type like short, byte, integer. * * @return long value of {@link SdkNumber}. */ @Override public long longValue() { if (numberValue instanceof Long) { return numberValue.longValue(); } if (stringValue != null) { return new BigDecimal(stringValue).longValue(); } return valueOf(numberValue).longValue(); } /** * Gets the float value of the {@link SdkNumber}. * If we do a floatValue() for {@link SdkNumber} constructed from * double, BigDecimal, BigInteger number type then it * may result in loss of magnitude and a loss of precision. * Precision is not lost while getting a {@link SdkNumber} which was constructed from * precision type like short, byte, integer, long. * * @return long value of {@link SdkNumber}. */ @Override public float floatValue() { if (numberValue instanceof Float) { return numberValue.floatValue(); } if (numberValue != null) { return valueOf(numberValue).floatValue(); } return new BigDecimal(stringValue).floatValue(); } /** * Gets the double value of the {@link SdkNumber}. * If we do a doubleValue() for {@link SdkNumber} constructed from BigDecimal, BigInteger number type then it * may result in loss of magnitude and a loss of precision. * Precision is not lost while getting a {@link SdkNumber} which was constructed from * precision type like short, byte, integer, long, float. * * @return long value of {@link SdkNumber}. */ @Override public double doubleValue() { if (numberValue instanceof Double) { return numberValue.doubleValue(); } if (numberValue != null) { return valueOf(numberValue).doubleValue(); } return new BigDecimal(stringValue).doubleValue(); } /** * Gets the bigDecimalValue of the {@link SdkNumber}. * Precision is not lost in this case. * However bigDecimalValue cannot be performed on * a {{@link SdkNumber}} constructed from Float/Double Nan/Infinity. * * @return BigDecimal value of {@link SdkNumber} * @throws NumberFormatException Exception in thrown if a {@link SdkNumber} was constructed asNan/Infinte number * of Double/FLoat type.Since we cannot convert NaN/Infinite numbers to BigDecimal. */ public BigDecimal bigDecimalValue() { if (stringValue != null) { return new BigDecimal(stringValue); } if (numberValue instanceof BigDecimal) { return (BigDecimal) numberValue; } if (isNumberValueNaN(numberValue) || isNumberValueInfinite(numberValue)) { throw new NumberFormatException("Nan or Infinite Number can not be converted to BigDecimal."); } else { return valueInBigDecimal(numberValue); } } /** * Gets the String value of the {@link SdkNumber}. * * @return the stringValue */ public String stringValue() { return stringValue != null ? stringValue : numberValue.toString(); } @Override public String toString() { return stringValue != null ? stringValue : numberValue.toString(); } @Override public boolean equals(Object o) { if (this == o) { return true; } if (!(o instanceof SdkNumber)) { return false; } SdkNumber sdkNumber = (SdkNumber) o; return Objects.equals(stringValue(), sdkNumber.stringValue()); } @Override public int hashCode() { return Objects.hashCode(stringValue()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy