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

com.anrisoftware.globalpom.math.measurement.AbstractValue Maven / Gradle / Ivy

/*
 * Copyright 2013-2025 Erwin Müller 
 *
 * 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.anrisoftware.globalpom.math.measurement;

/*-
 * #%L
 * Global POM Utilities :: Math
 * %%
 * Copyright (C) 2013 - 2018 Advanced Natural Research Institute
 * %%
 * 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.
 * #L%
 */

import static com.anrisoftware.globalpom.math.measurement.RoundToSignificantFigures.roundToDecimal;
import static com.anrisoftware.globalpom.math.measurement.RoundToSignificantFigures.roundToSignificant;
import static java.lang.Math.min;
import static org.apache.commons.math3.util.FastMath.max;

import java.io.Serializable;
import java.math.BigInteger;
import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.text.ParsePosition;
import java.util.Locale;

import jakarta.inject.Inject;

import org.apache.commons.lang3.builder.HashCodeBuilder;
import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.math3.util.FastMath;

import com.anrisoftware.globalpom.math.format.measurement.ValueFormat;
import com.anrisoftware.globalpom.math.format.measurement.ValueFormatFactory;
import com.anrisoftware.globalpom.math.math.MathUtils;

/**
 * Implements exact value's operators.
 *
 * @author Erwin Mueller, [email protected]
 * @since 2.4
 */
@SuppressWarnings("serial")
public abstract class AbstractValue implements Value, Serializable {

    private static final double DEFAULT_DIV = 3.0;

    private final BigInteger mantissa;

    private final int significant;

    private final double uncertainty;

    private final int decimal;

    private final int order;

    @Inject
    protected transient ValueFormatFactory valueFormatFactory;

    protected transient ValueFactory valueFactory;

    protected AbstractValue(Value value, ValueFactory valueFactory) {
        this(value.getMantissa(), value.getOrder(), value.getSignificant(),
                value.getDecimal(), value.getUncertainty(), valueFactory);
    }

    protected AbstractValue(long mantissa, int order, int sig, int dec,
            double unc, ValueFactory valueFactory) {
        this.mantissa = BigInteger.valueOf(mantissa);
        this.uncertainty = unc;
        this.order = order;
        this.significant = sig;
        this.decimal = dec;
        this.valueFactory = valueFactory;
    }

    protected AbstractValue(BigInteger mantissa, int order, int sig, int dec,
            double unc, ValueFactory valueFactory) {
        this.mantissa = mantissa;
        this.uncertainty = unc;
        this.order = order;
        this.significant = sig;
        this.decimal = dec;
        this.valueFactory = valueFactory;
    }

    @Override
    public BigInteger getMantissa() {
        return mantissa;
    }

    @Override
    public int getOrder() {
        return order;
    }

    @Override
    public int getSignificant() {
        return significant;
    }

    @Override
    public double getUncertainty() {
        if (isExact()) {
            return Double.NaN;
        }
        return uncertainty;
    }

    @Override
    public double getRoundedUncertainty() {
        if (isExact()) {
            return Double.NaN;
        }
        return roundToDecimal(uncertainty, getAbsDec());
    }

    @Override
    public int getDecimal() {
        return decimal;
    }

    @Override
    public boolean isExact() {
        return Double.isNaN(uncertainty);
    }

    @Override
    public double getValue() {
        return MathUtils.calculateValue(mantissa, decimal);
    }

    @Override
    public double getRoundedValue() {
        return roundedValue(getSignificant(), getAbsDec());
    }

    @Override
    public double roundedValue(int sig, int dec) {
        double value = getValue();
        value = roundToSignificant(value, sig);
        value = roundToDecimal(value, dec);
        return value;
    }

    @Override
    public Value getMinValue() {
        return minValue(DEFAULT_DIV);
    }

    @Override
    public Value minValue(double deviation) {
        if (isExact()) {
            return this;
        } else {
            return this.sub(deviation * getUncertainty());
        }
    }

    @Override
    public Value getMaxValue() {
        return maxValue(DEFAULT_DIV);
    }

    @Override
    public Value maxValue(double deviation) {
        if (isExact()) {
            return this;
        } else {
            return add(deviation * getUncertainty());
        }
    }

    @Override
    public Value valueOf(BigInteger mantissa, int order, int sig, int dec,
            double unc) {
        return createValue(mantissa, order, sig, dec, unc);
    }

    @Override
    public Value valueOf(double value, int sig, int dec, double unc) {
        return createValue(value, sig, dec, unc);
    }

    @Override
    public Value valueOf(double value, int dec, double unc) {
        return createValue(value, dec, unc);
    }

    @Override
    public Value add(Value addend) {
        double value = this.getValue() + addend.getValue();
        double unc = addUncertainty(addend, value);
        int sig = min(significant, addend.getSignificant());
        int dec = max(decimal, addend.getDecimal());
        double rvalue = roundToDecimal(value, getAbsDec());
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value add(double addend) {
        double value = this.getValue() + addend;
        double unc = addUncertainty(addend, value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToDecimal(value, getAbsDec());
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value plus(Value addend) {
        return add(addend);
    }

    @Override
    public Value plus(double addend) {
        return add(addend);
    }

    /**
     * Adds the uncertainty of this value and the addend.
     *
     * @param addend
     *            the {@link Value} addend.
     *
     * @param sum
     *            the value of the sum.
     *
     * @return the added uncertainly or {@link Double#NaN} if the value is
     *         exact.
     */
    protected abstract double addUncertainty(Value addend, double sum);

    /**
     * Adds the uncertainty of this value and the addend.
     *
     * @param addend
     *            the {@link Double} addend.
     *
     * @param sum
     *            the value of the sum.
     *
     * @return the added uncertainly or {@link Double#NaN} if the value is
     *         exact.
     */
    protected abstract double addUncertainty(double addend, double sum);

    @Override
    public Value sub(Value subtrahend) {
        double value = this.getValue() - subtrahend.getValue();
        double unc = subUncertainty(subtrahend, value);
        int sig = min(significant, subtrahend.getSignificant());
        int dec = max(decimal, subtrahend.getDecimal());
        double rvalue = roundToDecimal(value, getAbsDec());
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value sub(double subtrahend) {
        double value = this.getValue() - subtrahend;
        double unc = subUncertainty(subtrahend, value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToDecimal(value, getAbsDec());
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value minus(Value subtrahend) {
        return sub(subtrahend);
    }

    @Override
    public Value minus(double subtrahend) {
        return sub(subtrahend);
    }

    /**
     * Subtracts the uncertainty of this value and the subtrahend.
     *
     * @param subtrahend
     *            the {@link Value} subtrahend.
     *
     * @param diff
     *            the value of the difference.
     *
     * @return the subtracted uncertainly.
     */
    protected abstract double subUncertainty(Value subtrahend, double diff);

    /**
     * Subtracts the uncertainty of this value and the subtrahend.
     *
     * @param subtrahend
     *            the {@link Double} subtrahend.
     *
     * @param diff
     *            the value of the difference.
     *
     * @return the subtracted uncertainly.
     */
    protected abstract double subUncertainty(double subtrahend, double diff);

    @Override
    public Value mul(Value factor) {
        double value = this.getValue() * factor.getValue();
        double unc = mulUncertainty(factor, value);
        int sig = min(significant, factor.getSignificant());
        int dec = max(decimal, factor.getDecimal());
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value mul(double factor) {
        double value = this.getValue() * factor;
        double unc = mulUncertainty(factor, value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value multiply(Value factor) {
        return mul(factor);
    }

    @Override
    public Value multiply(double factor) {
        return mul(factor);
    }

    /**
     * Multiplies the uncertainty of this value and the factor.
     *
     * @param factor
     *            the {@link Value} factor.
     *
     * @param product
     *            the value of the product.
     *
     * @return the multiplied uncertainly.
     */
    protected abstract double mulUncertainty(Value factor, double product);

    /**
     * Multiplies the uncertainty of this value and the factor.
     *
     * @param factor
     *            the {@link Double} factor.
     *
     * @param product
     *            the value of the product.
     *
     * @return the multiplied uncertainly.
     */
    protected abstract double mulUncertainty(double factor, double product);

    @Override
    public Value div(Value divisor) {
        double value = this.getValue() / divisor.getValue();
        double unc = divUncertainty(divisor, value);
        int sig = min(significant, divisor.getSignificant());
        int dec = max(decimal, divisor.getDecimal());
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value div(double divisor) {
        double value = this.getValue() / divisor;
        double unc = divUncertainty(divisor, value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value divNum(Value numerator) {
        double value = numerator.getValue() / this.getValue();
        double unc = divUncertainty(numerator, value);
        int sig = min(significant, numerator.getSignificant());
        int dec = max(decimal, numerator.getDecimal());
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    @Override
    public Value divNum(double numerator) {
        double value = numerator / this.getValue();
        double unc = divUncertainty(numerator, value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    /**
     * Divides the uncertainty of this value and the divisor.
     *
     * @param divisor
     *            the {@link Value} divisor.
     *
     * @param quotient
     *            the value of the quotient.
     *
     * @return the divided uncertainly.
     */
    protected abstract double divUncertainty(Value divisor, double quotient);

    /**
     * Divides the uncertainty of this value and the divisor.
     *
     * @param divisor
     *            the {@link Double} divisor.
     *
     * @param quotient
     *            the value of the quotient.
     *
     * @return the divided uncertainly.
     */
    protected abstract double divUncertainty(double divisor, double quotient);

    @Override
    public Value reciprocal() {
        double value = 1.0 / this.getValue();
        double unc = reciprocalUncertainty(value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    /**
     * Calculates the uncertainty of this value as reciprocal.
     *
     * @param value
     *            the value.
     *
     * @return the reciprocal uncertainly.
     */
    protected abstract double reciprocalUncertainty(double value);

    @Override
    public Value log() {
        double value = FastMath.log(this.getValue());
        double unc = logUncertainty(value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    /**
     * Logarithm the uncertainty of this value.
     *
     * @param exponent
     *            the value of the exponent.
     *
     * @return the logarithm uncertainly.
     */
    protected abstract double logUncertainty(double exponent);

    @Override
    public Value exp() {
        double value = FastMath.exp(this.getValue());
        double unc = expUncertainty(value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToDecimal(value, getAbsDec());
        return valueOf(rvalue, sig, dec, unc);
    }

    /**
     * Power to the basis of e the uncertainty of this value.
     *
     * @param power
     *            the value of the power.
     *
     * @return the power uncertainly.
     */
    protected abstract double expUncertainty(double power);

    @Override
    public Value abs() {
        double value = FastMath.abs(this.getValue());
        double unc = absUncertainty(value);
        int sig = significant;
        int dec = decimal;
        double rvalue = roundToSignificant(value, sig);
        return valueOf(rvalue, sig, dec, unc);
    }

    /**
     * Absolute uncertainty of this value.
     *
     * @param value
     *            the absolute value.
     *
     * @return the uncertainly.
     */
    protected abstract double absUncertainty(double value);

    @Override
    public Value square() {
        return this.mul(this);
    }

    @Override
    public Value cube() {
        return this.square().mul(this);
    }

    /**
     * Creates a new value with the specified uncertainty.
     *
     * @param m
     *            the significant digits of the value.
     *
     * @param order
     *            the order of the value.
     *
     * @param sig
     *            the significant figures of the value.
     *
     * @param dec
     *            the least significant decimal.
     *
     * @param unc
     *            the uncertainty {@link Value}.
     *
     * @return the {@link Value}.
     */
    protected Value createValue(BigInteger m, int order, int sig, int dec,
            double unc) {
        return valueFactory.create(m, order, sig, dec, unc, valueFactory);
    }

    @Override
    public int compareTo(Value rhs) {
        return Double.compare(this.getValue(), rhs.getValue());
    }

    @Override
    public boolean equals(Object obj) {
        if (obj == null) {
            return false;
        }
        if (obj == this) {
            return true;
        }
        if (!(obj instanceof Value)) {
            return false;
        }
        Value rhs = (Value) obj;
        return equalExact(this, rhs);
    }

    @Override
    public boolean isConsistent(Value rhs) {
        return rhs == null ? false : rhs.isExact() ? equalExact(this, rhs)
                : equalsUncertain(rhs);
    }

    private boolean equalsUncertain(Value rhs) {
        double delta = FastMath.abs(this.getValue() - rhs.getValue());
        double deltaunc = this.getUncertainty() + rhs.getUncertainty();
        return delta <= deltaunc;
    }

    private boolean equalExact(Value lhs, Value rhs) {
        return lhs.getMantissa().equals(rhs.getMantissa())
                && lhs.getDecimal() == rhs.getDecimal();
    }

    @Override
    public int hashCode() {
        return new HashCodeBuilder().append(getMantissa()).append(getDecimal())
                .hashCode();
    }

    @Override
    public String toString() {
        return new ToStringBuilder(this).append(MANTISSA_FIELD, mantissa)
                .append(ORDER_FIELD, order).append(SIG_FIELD, significant)
                .append(DEC_FIELD, decimal).append(UNC_FIELD, uncertainty)
                .toString();
    }

    private Value createValue(double value, int sig, int dec, double unc) {
        double avalue = FastMath.abs(value);
        StringBuilder pattern = new StringBuilder("0");
        pattern.append('.');
        if (avalue > 0) {
            sig--;
        }
        for (int i = 0; i < sig; i++) {
            pattern.append('0');
        }
        pattern.append("E0");
        String svalue;
        Locale locale = Locale.ENGLISH;
        DecimalFormatSymbols symbols = DecimalFormatSymbols.getInstance(locale);
        svalue = new DecimalFormat(pattern.toString(), symbols).format(value);
        if (!Double.isNaN(unc)) {
            String uvalue = new DecimalFormat(String.format("(%s)",
                    pattern.toString()), symbols).format(unc);
            svalue += uvalue;
        }
        ValueFormat format = valueFormatFactory.create(locale, valueFactory);
        ParsePosition pos = new ParsePosition(0);
        Value vv = format.parse(svalue, pos);
        return vv;
    }

    private Value createValue(double value, int dec, double unc) {
        String svalue = Double.toString(value);
        Locale locale = Locale.ENGLISH;
        ValueFormat format;
        format = valueFormatFactory.create(locale, valueFactory);
        format.setDecimal(dec);
        ParsePosition pos = new ParsePosition(0);
        Value vv = format.parse(svalue, unc, pos);
        return vv;
    }

    private int getAbsDec() {
        return decimal < 0 ? -1 * decimal : 0;
    }

    private static final String UNC_FIELD = "unc";

    private static final String DEC_FIELD = "dec";

    private static final String SIG_FIELD = "sig";

    private static final String ORDER_FIELD = "order";

    private static final String MANTISSA_FIELD = "mantissa";

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy