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

com.github.richardballard.arbeeutils.numeric.Count Maven / Gradle / Ivy

/*
 * (C) Copyright 2016 Richard Ballard.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License 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 com.github.richardballard.arbeeutils.numeric;

import com.github.richardballard.arbeecoretypes.numeric.BigNumber;
import net.jcip.annotations.Immutable;
import org.jetbrains.annotations.NotNull;

import java.math.BigDecimal;
import java.math.BigInteger;
import java.math.RoundingMode;

/**
 * This class represents a zero based count.  For any method that creates an instance (e.g. {@link #valueOf(BigInteger, OnExceedBoundary)}
 * there is an {@code onExceedBoundary} parameter which indicates what should happen if the result would be negative
 * (which doesn't make sense for a count).
 */
@Immutable
public class Count extends BigNumber {

    private static final long serialVersionUID = 8108782711353619422L;

    public static final Count ZERO = new Count(BigInteger.ZERO);
    public static final Count MIN_VALUE = ZERO;

    public enum OnExceedBoundary {
        /**
         * e.g. if the value is > MIN_VALUE then set to MIN_VALUE
         */
        USE_BOUNDARY_VALUE,

        /**
         * throw an exception
         */
        THROW
    }

    @NotNull
    public static Count valueOf(@NotNull final BigInteger value,
                                @NotNull final OnExceedBoundary onExceedBoundary) {
        assert value != null;
        assert onExceedBoundary != null;

        final Count result;

        final int comparisonToZero = value.compareTo(BigInteger.ZERO);
        if(comparisonToZero == 0) {
            result = ZERO;
        }
        else if(comparisonToZero > 0) {
            result = new Count(value);
        }
        else {
            if(onExceedBoundary == OnExceedBoundary.THROW) {
                throw new IllegalArgumentException("value must be >= 0, not " + value);
            }
            else if(onExceedBoundary == OnExceedBoundary.USE_BOUNDARY_VALUE) {
                result = ZERO;
            }
            else {
                throw new IllegalStateException("unknown onExceedBoundary - " + onExceedBoundary);
            }
        }

        return result;
    }

    @NotNull
    private static BigInteger asBigInteger(final @NotNull BigNumber value) {
        assert value != null;

        return value.bigIntegerValue();
    }

    @NotNull
    private static BigInteger asBigInteger(final long value) {

        return BigInteger.valueOf(value);
    }

    @NotNull
    private static BigInteger asBigInteger(@NotNull final Number value) {
        assert value != null;

        return asBigInteger(value.longValue());
    }


    @NotNull
    public static Count valueOf(final @NotNull BigNumber value,
                                @NotNull final OnExceedBoundary onExceedBoundary) {
        assert value != null;
        assert onExceedBoundary != null;

        return valueOf(asBigInteger(value),
                       onExceedBoundary);
    }

    @NotNull
    public static Count valueOf(final long value,
                                @NotNull final OnExceedBoundary onExceedBoundary) {
        assert onExceedBoundary != null;

        return valueOf(asBigInteger(value),
                       onExceedBoundary);
    }

    /**
     *
     * @param value this will be truncated to an integral value (not rounded)
     */
    @NotNull
    public static Count valueOf(@NotNull final Number value,
                                @NotNull final OnExceedBoundary onExceedBoundary) {
        assert value != null;
        assert onExceedBoundary != null;

        return valueOf(asBigInteger(value),
                       onExceedBoundary);
    }

    @NotNull
    private final BigInteger value;

    private Count(@NotNull final BigInteger value) {
        assert value != null;
        assert value.compareTo(BigInteger.ZERO) >= 0;

        this.value = value;
    }

    /**
     *
     * @return the value which will be >= 0
     */
    @NotNull
    public BigInteger getValue() {
        return value;
    }

    @NotNull
    @Override
    public BigInteger bigIntegerValue() {
        return value;
    }

    @NotNull
    @Override
    public BigDecimal bigDecimalValue() {
        return new BigDecimal(value);
    }

    /**
     *
     * @return the value which will be >= 0 (cast to int)
     */
    @Override
    public int intValue() {
        return value.intValue();
    }

    /**
     *
     * @return the value which will be >= 0
     */
    @Override
    public long longValue() {
        return value.longValue();
    }

    /**
     *
     * @return the value which will be >= 0 (cast to float)
     */
    @Override
    public float floatValue() {
        return value.floatValue();
    }

    /**
     *
     * @return the value which will be >= 0 (cast to double)
     */
    @Override
    public double doubleValue() {
        return value.doubleValue();
    }

    @NotNull
    public Count plus(@NotNull final BigInteger augend,
                      @NotNull final OnExceedBoundary onExceedBoundary) {
        assert augend != null;
        assert onExceedBoundary != null;

        return valueOf(value.add(augend),
                       onExceedBoundary);
    }

    @NotNull
    public Count plus(final @NotNull BigNumber augend,
                      @NotNull final OnExceedBoundary onExceedBoundary) {
        assert augend != null;
        assert onExceedBoundary != null;

        return plus(asBigInteger(augend),
                    onExceedBoundary);
    }

    @NotNull
    public Count plus(final long augend,
                      @NotNull final OnExceedBoundary onExceedBoundary) {
        assert onExceedBoundary != null;

        return plus(asBigInteger(augend),
                    onExceedBoundary);
    }

    /**
     *
     * @param augend this will be truncated to an integral value (not rounded)
     */
    @NotNull
    public Count plus(@NotNull final Number augend,
                      @NotNull final OnExceedBoundary onExceedBoundary) {
        assert augend != null;
        assert onExceedBoundary != null;

        return plus(asBigInteger(augend),
                    onExceedBoundary);
    }

    @NotNull
    public Count minus(@NotNull final BigInteger subtrahend,
                       @NotNull final OnExceedBoundary onExceedBoundary) {
        assert subtrahend != null;
        assert onExceedBoundary != null;

        return valueOf(value.subtract(subtrahend),
                       onExceedBoundary);
    }

    @NotNull
    public Count minus(final @NotNull BigNumber subtrahend,
                       @NotNull final OnExceedBoundary onExceedBoundary) {
        assert subtrahend != null;
        assert onExceedBoundary != null;

        return minus(asBigInteger(subtrahend),
                     onExceedBoundary);
    }

    @NotNull
    public Count minus(final long augend,
                       @NotNull final OnExceedBoundary onExceedBoundary) {
        assert onExceedBoundary != null;

        return minus(asBigInteger(augend),
                     onExceedBoundary);
    }

    /**
     *
     * @param subtrahend this will be truncated to an integral value (not rounded)
     */
    @NotNull
    public Count minus(@NotNull final Number subtrahend,
                       @NotNull final OnExceedBoundary onExceedBoundary) {
        assert subtrahend != null;
        assert onExceedBoundary != null;

        return minus(asBigInteger(subtrahend),
                     onExceedBoundary);
    }


    /**
     * If the result (before conversion to a {@link Count} instance) has a fractional component then this will be
     * truncated.
     */
    @NotNull
    public Count multipliedBy(@NotNull final BigDecimal multiplicand,
                              @NotNull final OnExceedBoundary onExceedBoundary) {
        assert multiplicand != null;
        assert onExceedBoundary != null;

        final BigDecimal result = new BigDecimal(value).multiply(multiplicand);

        return valueOf(result.toBigInteger(),
                       onExceedBoundary);
    }

    @NotNull
    public Count multipliedBy(@NotNull final BigInteger multiplicand,
                              @NotNull final OnExceedBoundary onExceedBoundary) {
        assert multiplicand != null;
        assert onExceedBoundary != null;

        return valueOf(value.multiply(multiplicand),
                       onExceedBoundary);
    }

    /**
     * If the result (before conversion to a {@link Count} instance) has a fractional component then this will be
     * truncated.
     */
    @NotNull
    public Count multipliedBy(@NotNull final BigNumber multiplicand,
                              @NotNull final OnExceedBoundary onExceedBoundary) {
        assert multiplicand != null;
        assert onExceedBoundary != null;

        return multipliedBy(multiplicand.bigDecimalValue(),
                            onExceedBoundary);
    }

    @NotNull
    public Count multipliedBy(final long multiplicand,
                              @NotNull final OnExceedBoundary onExceedBoundary) {
        assert onExceedBoundary != null;

        return multipliedBy(asBigInteger(multiplicand),
                            onExceedBoundary);
    }

    /**
     * If the result (before conversion to a {@link Count} instance) has a fractional component then this will be
     * truncated.
     */
    @NotNull
    public Count multipliedBy(@NotNull final Number multiplicand,
                              @NotNull final OnExceedBoundary onExceedBoundary) {
        assert multiplicand != null;
        assert onExceedBoundary != null;

        return multipliedBy(BigDecimal.valueOf(multiplicand.doubleValue()),
                            onExceedBoundary);
    }

    /**
     * If the result (before conversion to a {@link Count} instance) has a fractional component then this will be
     * truncated.
     */
    @NotNull
    public Count dividedBy(@NotNull final BigDecimal divisor,
                           @NotNull final OnExceedBoundary onExceedBoundary) {
        assert divisor != null;
        assert onExceedBoundary != null;

        final BigDecimal result = new BigDecimal(value).divide(divisor,
                                                               10,      // make sure there is plenty of room in the scale so that the integer component is not rounded
                                                               RoundingMode.HALF_UP);

        return valueOf(result.toBigInteger(),
                       onExceedBoundary);
    }

    @NotNull
    public Count dividedBy(@NotNull final BigInteger divisor,
                           @NotNull final OnExceedBoundary onExceedBoundary) {
        assert divisor != null;
        assert onExceedBoundary != null;

        return valueOf(value.divide(divisor),
                       onExceedBoundary);
    }

    /**
     * If the result (before conversion to a {@link Count} instance) has a fractional component then this will be
     * truncated.
     */
    @NotNull
    public Count dividedBy(@NotNull final BigNumber divisor,
                           @NotNull final OnExceedBoundary onExceedBoundary) {
        assert divisor != null;
        assert onExceedBoundary != null;

        return dividedBy(divisor.bigDecimalValue(),
                         onExceedBoundary);
    }

    @NotNull
    public Count dividedBy(final long divisor,
                           @NotNull final OnExceedBoundary onExceedBoundary) {
        assert onExceedBoundary != null;

        return dividedBy(asBigInteger(divisor),
                         onExceedBoundary);
    }

    /**
     * If the result (before conversion to a {@link Count} instance) has a fractional component then this will be
     * truncated.
     */
    @NotNull
    public Count dividedBy(@NotNull final Number divisor,
                           @NotNull final OnExceedBoundary onExceedBoundary) {
        assert divisor != null;
        assert onExceedBoundary != null;

        return multipliedBy(BigDecimal.valueOf(divisor.doubleValue()),
                            onExceedBoundary);
    }

    @Override
    public boolean equals(final Object o) {
        if(this == o) {
            return true;
        }
        if(o == null || getClass() != o.getClass()) {
            return false;
        }

        final Count count = (Count) o;

        return value.equals(count.value);

    }

    @Override
    public int hashCode() {
        return value.hashCode();
    }

    @Override
    public String toString() {
        return "Count{" +
               "value=" + String.format("%,d", value) +
               '}';
    }
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy