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

org.monte.media.math.Rational Maven / Gradle / Ivy

The newest version!

package org.monte.media.math;

import static java.lang.Math.*;
import java.math.BigInteger;
import static org.monte.media.math.IntMath.*;


public class Rational extends Number {

    public static final Rational ONE = new Rational(1, 1,false);
    public static final Rational ZERO = new Rational(0, 1,false);
    public static final long serialVersionUID = 1L;
    private final long num;
    private final long den;

    public Rational(long numerator) {
        this(numerator, 1);
    }

    public Rational(long numerator, long denominator) {
        this(numerator, denominator, true);
    }

    private Rational(long numerator, long denominator, boolean reduceFraction) {
        if (numerator == 0) {

            denominator = 1;
        }

        if (denominator == 0) {

            numerator = (numerator > 0) ? 1 : -1;
        } else if (denominator < 0) {

            denominator = -denominator;
            numerator = -numerator;
        }

        if (reduceFraction) {
            long g = gcd(numerator, denominator);
            num = numerator / g;
            den = denominator / g;
        } else {
            num = numerator;
            den = denominator;
        }
    }

    private Rational(BigInteger numerator, BigInteger denominator, boolean reduceFraction) {
        if (numerator.equals(BigInteger.ZERO)) {

            denominator = BigInteger.ONE;
        }

        if (denominator.equals(BigInteger.ZERO)) {

            numerator = (numerator.compareTo(BigInteger.ZERO) > 0) ? BigInteger.ONE : BigInteger.ONE.negate();
        } else if (denominator.compareTo(BigInteger.ZERO) < 0) {

            denominator = denominator.negate();
            numerator = numerator.negate();
        }

        BigInteger numB, denB;
        if (reduceFraction) {
            BigInteger g = gcd(numerator, denominator);
            numB = numerator.divide(g);
            denB = denominator.divide(g);
        } else {
            numB = numerator;
            denB = denominator;
        }
        int bitLength = Math.max(numB.bitLength(), denB.bitLength());
        if (bitLength > 63) {
            numB = numB.shiftRight(bitLength - 63);
            denB = denB.shiftRight(bitLength - 63);
            if (numB.equals(BigInteger.ZERO)) {

                denB = BigInteger.ONE;
            }

            if (denB.equals(BigInteger.ZERO)) {

                numB = (numB.compareTo(BigInteger.ZERO) > 0) ? BigInteger.ONE : BigInteger.ONE.negate();

            }
        }
        num = numB.longValue();
        den = denB.longValue();
    }

    public Rational(Rational r) {
        this(r.num, r.den);
    }

    public long getNumerator() {
        return num;
    }

    public long getDenominator() {
        return den;
    }

    public Rational add(Rational that) {
        return add(that, true);
    }

    private Rational add(Rational that, boolean reduceFraction) {
        if (this.den == that.den) {

            return new Rational(this.num + that.num, this.den, reduceFraction);
        }


        long s = scm(this.den, that.den);
        Rational result = new Rational(
                this.num * (s / this.den) + that.num * (s / that.den),
                s, reduceFraction);

        return result;
    }

    
    public Rational subtract(Rational that) {
        return add(that.negate());
    }

    public Rational negate() {
        return valueOf(-num, den);
    }

    public Rational inverse() {
        return valueOf(den, num, false);
    }

    
    public Rational floor(long d) {
        if (d == den) {
            return valueOf(num, den);
        }
        long s = scm(this.den, d);

        if (s == d) {
            return valueOf(num * s / den, d);
        } else if (s == den) {
            return valueOf(num * d / den, d);
        } else {
            return valueOf(num * d / den, d);
        }
    }

    
    public Rational ceil(long d) {
        if (d == den) {
            return valueOf(num, den);
        }
        long s = scm(this.den, d);

        if (s == d) {
            return valueOf((num * s + den - 1) / den, d);
        } else if (s == den) {
            return valueOf((num * d + den - 1) / den, d);
        } else {
            return valueOf((num * d + den - 1) / den, d);
        }
    }

    public Rational multiply(Rational that) {
        if (abs(this.num) < Integer.MAX_VALUE
                && abs(this.den) < Integer.MAX_VALUE
                && abs(that.num) < Integer.MAX_VALUE
                && abs(that.den) < Integer.MAX_VALUE) {
            return valueOf(this.num * that.num,
                    this.den * that.den);
        } else {
            return new Rational(
                    BigInteger.valueOf(this.num).multiply(BigInteger.valueOf(that.num)),
                    BigInteger.valueOf(this.den).multiply(BigInteger.valueOf(that.den)),
                    true);
        }
    }

    public Rational multiply(long integer) {
        if (integer==0) {
            return ZERO;
        } else if (this.den % integer == 0) {
            return valueOf(
                    this.num,
                    this.den / integer);
        } else if (abs(this.num) < Integer.MAX_VALUE
                && abs(integer) < Integer.MAX_VALUE) {
            return valueOf(
                    this.num * integer,
                    this.den);
        } else {
            return new Rational(
                    BigInteger.valueOf(this.num).multiply(BigInteger.valueOf(integer)),
                    BigInteger.valueOf(this.den), true);
        }
    }

    public Rational divide(Rational that) {
        if (abs(this.num) < Integer.MAX_VALUE
                && abs(this.den) < Integer.MAX_VALUE
                && abs(that.num) < Integer.MAX_VALUE
                && abs(that.den) < Integer.MAX_VALUE) {
            return valueOf(this.num * that.den,
                    this.den * that.num);
        } else {
            return valueOf(
                    BigInteger.valueOf(this.num).multiply(BigInteger.valueOf(that.den)),
                    BigInteger.valueOf(this.den).multiply(BigInteger.valueOf(that.num)),
                    true);
        }
    }

    @Override
    public String toString() {

        if (num == 0) {
            return "0";
        } else if (den == 1) {
            return Long.toString(num);
        } else {
            return num + "/" + den;
            
        }
    }

    public String toDescriptiveString() {
        long gcd = IntMath.gcd(num, den);
        if (gcd == 0 || num == 0) {
            return num + "/" + den + " = " + 0;
        } else if (gcd == den) {
            return num + "/" + den + " = " + Long.toString(num / den);
        } else {
            return num + "/" + den + " ≈ " + ((float) num / den);
        }
    }

    @Override
    public int intValue() {
        return (int) (num / den);
    }

    @Override
    public long longValue() {
        return num / den;
    }

    @Override
    public float floatValue() {
        return (float) num / (float) den;
    }

    @Override
    public double doubleValue() {
        return (double) num / (double) den;
    }

    @Override
    public boolean equals(Object obj) {
        if (obj == null) {
            return false;
        }
        if (getClass() != obj.getClass()) {
            return false;
        }
        final Rational that = (Rational) obj;

        return compareTo(that) == 0;
    }

    
    public int compareTo(Rational that) {


        if (this.den == that.den) {
            if (this.num < that.num) {
                return -1;
            } else if (this.num > that.num) {
                return 1;
            } else {
                return 0;
            }
        }


        if (abs(this.num) < Integer.MAX_VALUE
                && abs(this.den) < Integer.MAX_VALUE
                && abs(that.num) < Integer.MAX_VALUE
                && abs(that.den) < Integer.MAX_VALUE) {
            long lhs = this.num * that.den;
            long rhs = this.den * that.num;
            if (lhs < rhs) {
                return -1;
            } else if (lhs > rhs) {
                return 1;
            } else {
                return 0;
            }
        }


        BigInteger lhs;
        BigInteger rhs;
        lhs = BigInteger.valueOf(this.num).multiply(BigInteger.valueOf(that.den));
        rhs = BigInteger.valueOf(this.den).multiply(BigInteger.valueOf(that.num));

        return lhs.compareTo(rhs);
    }

    @Override
    public int hashCode() {
        return (int) ((num ^ (num >>> 32))
                ^ (den ^ (den >>> 32)));

    }

    public static Rational max(Rational a, Rational b) {
        return (a.compareTo(b) >= 0) ? a : b;
    }

    public static Rational min(Rational a, Rational b) {
        return (a.compareTo(b) <= 0) ? a : b;
    }

    public boolean isZero() {
        return num == 0;
    }

    public boolean isLessOrEqualZero() {
        return num <= 0;
    }

    public static Rational valueOf(double d) {
        if (d == 0) {
            return valueOf(0, 1);
        }
        if (abs(d) > Integer.MAX_VALUE) {
            throw new IllegalArgumentException("Value " + d + " is too big.");
        }
        if (Double.isInfinite(d)) {
            return valueOf((long) signum(d), 0);
        }
        if (Double.isNaN(d)) {
            return valueOf(0, 1);
        }
        return toRational(d, Integer.MAX_VALUE, 100);
    }

    public static Rational valueOf(long num, long den) {
        return valueOf(num, den, true);
    }

    private static Rational valueOf(long num, long den, boolean reduceFraction) {
        if (num == den) {
            return ONE;
        }
        if (num == 0) {
            return ZERO;
        }
        return new Rational(num, den, reduceFraction);
    }

    public static Rational valueOf(BigInteger num, BigInteger den) {
        return valueOf(num, den, true);
    }

    private static Rational valueOf(BigInteger num, BigInteger den, boolean reduceFraction) {
        if (num.equals(den)) {
            return ONE;
        }
        if (num.equals(BigInteger.ZERO)) {
            return ZERO;
        }
        return new Rational(num, den, reduceFraction);
    }

    
    private static Rational toRational(double x, double limit, int iterations) {
        double intpart = Math.floor(x);
        double fractpart = x - intpart;
        double d = 1.0 / fractpart;
        long left = (long) intpart;
        if (d > limit || iterations == 0) {
            return valueOf(left, 1, false);
        } else {
            return valueOf(left, 1, false).add(toRational(d, limit * 0.1, iterations - 1).inverse(), false);
        }
    }

    public Rational round(long d) {
        if (d == den) {
            return valueOf(num, den);
        }

        Rational fl = floor(d);
        Rational diffFl = subtract(fl);

        if (diffFl.isZero()) {
            return fl;
        }

        Rational cl = ceil(d);
        Rational diffCl = subtract(cl);
        if (diffCl.isZero()) {
            return cl;
        }

        if (diffFl.isNegative()) {
            diffFl = diffFl.negate();
        }
        if (diffCl.isNegative()) {
            diffCl = diffCl.negate();
        }
        return diffFl.compareTo(diffCl) <= 0 ? fl : cl;
    }

    private boolean isNegative() {
        return num < 0;
    }

    
    public static Rational valueOf(String str) {
        int p = str.indexOf('/');
        if (p != -1) {
            return valueOf(Long.valueOf(str.substring(0, p)), Long.valueOf(str.substring(p + 1)));
        }
        try {
            return valueOf(Long.valueOf(str));
        } catch (NumberFormatException e) {
            return valueOf(Double.valueOf(str));
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy