Cara terbaik untuk merepresentasikan pecahan di Jawa?

100

Saya mencoba bekerja dengan pecahan di Jawa.

Saya ingin menerapkan fungsi aritmatika. Untuk ini, pertama-tama saya akan membutuhkan cara untuk menormalkan fungsi. Saya tahu saya tidak dapat menambahkan 1/6 dan 1/2 sampai saya memiliki penyebut yang sama. Saya harus menambahkan 1/6 dan 3/6. Pendekatan yang naif akan meminta saya menambahkan 2/12 dan 6/12 lalu mengurangi. Bagaimana saya bisa mencapai penyebut umum dengan penalti performa paling kecil? Algoritma apa yang terbaik untuk ini?


Versi 8 (terima kasih kepada hstoerr ):

Perbaikan meliputi:

  • metode equals () sekarang konsisten dengan metode bandingkanTo ()
final class Fraction extends Number {
    private int numerator;
    private int denominator;

    public Fraction(int numerator, int denominator) {
        if(denominator == 0) {
            throw new IllegalArgumentException("denominator is zero");
        }
        if(denominator < 0) {
            numerator *= -1;
            denominator *= -1;
        }
        this.numerator = numerator;
        this.denominator = denominator;
    }

    public Fraction(int numerator) {
        this.numerator = numerator;
        this.denominator = 1;
    }

    public int getNumerator() {
        return this.numerator;
    }

    public int getDenominator() {
        return this.denominator;
    }

    public byte byteValue() {
        return (byte) this.doubleValue();
    }

    public double doubleValue() {
        return ((double) numerator)/((double) denominator);
    }

    public float floatValue() {
        return (float) this.doubleValue();
    }

    public int intValue() {
        return (int) this.doubleValue();
    }

    public long longValue() {
        return (long) this.doubleValue();
    }

    public short shortValue() {
        return (short) this.doubleValue();
    }

    public boolean equals(Fraction frac) {
        return this.compareTo(frac) == 0;
    }

    public int compareTo(Fraction frac) {
        long t = this.getNumerator() * frac.getDenominator();
        long f = frac.getNumerator() * this.getDenominator();
        int result = 0;
        if(t>f) {
            result = 1;
        }
        else if(f>t) {
            result = -1;
        }
        return result;
    }
}

Saya telah menghapus semua versi sebelumnya. Terima kasih saya kepada:

eleven81
sumber
33
Buang kode, gunakan Apache Commons :) commons.apache.org/math/userguide/fraction.html
Patrick
3
Komentar Patrick layak mendapat +1, seandainya itu diposkan sebagai jawaban. Dalam banyak kasus, itulah jawaban yang benar; "tahu dan gunakan perpustakaan", seperti yang dikatakan Java Efektif. Pertanyaan aslinya jelas dan berguna juga.
Jonik
Perhatikan bahwa Anda menerima jawaban saya .. jika Anda benar-benar menggunakan kode itu dan menemukan masalah dengan kode itu atau apa pun yang kurang, beri tahu saya! email saya dari situs web saya: vacant-nebula.com/contact/kip
Kip
Saya sarankan Anda mengedit metode "CompareTo", dan mentransmisikan "this.getNumerator ()" jauh sebelum perkalian. Jika tidak, kode tersebut masih cenderung meluap. Juga saya pikir akan menyenangkan untuk menerapkan <Fraksi> yang sebanding, karena Anda telah menerapkan metode bandingkanTo.
Hosam Aly
Dan karena Anda telah melangkah sejauh ini, mungkin berguna untuk mengimplementasikan persamaan dan kode hash juga.
Hosam Aly

Jawaban:

65

Kebetulan saya menulis kelas BigFraction belum lama ini, untuk masalah Project Euler . Itu menyimpan pembilang dan penyebut BigInteger, sehingga tidak akan pernah meluap. Tapi itu akan menjadi anak laki-laki lambat untuk banyak operasi yang Anda tahu tidak akan pernah meluap .. bagaimanapun, gunakan jika Anda menginginkannya. Aku sangat ingin memamerkan ini. :)

Sunting : Versi terbaru dan terhebat dari kode ini, termasuk pengujian unit sekarang dihosting di GitHub dan juga tersedia melalui Maven Central . Saya meninggalkan kode asli saya di sini sehingga jawaban ini bukan hanya tautan ...


import java.math.*;

/**
 * Arbitrary-precision fractions, utilizing BigIntegers for numerator and
 * denominator.  Fraction is always kept in lowest terms.  Fraction is
 * immutable, and guaranteed not to have a null numerator or denominator.
 * Denominator will always be positive (so sign is carried by numerator,
 * and a zero-denominator is impossible).
 */
public final class BigFraction extends Number implements Comparable<BigFraction>
{
  private static final long serialVersionUID = 1L; //because Number is Serializable
  private final BigInteger numerator;
  private final BigInteger denominator;

  public final static BigFraction ZERO = new BigFraction(BigInteger.ZERO, BigInteger.ONE, true);
  public final static BigFraction ONE = new BigFraction(BigInteger.ONE, BigInteger.ONE, true);

  /**
   * Constructs a BigFraction with given numerator and denominator.  Fraction
   * will be reduced to lowest terms.  If fraction is negative, negative sign will
   * be carried on numerator, regardless of how the values were passed in.
   */
  public BigFraction(BigInteger numerator, BigInteger denominator)
  {
    if(numerator == null)
      throw new IllegalArgumentException("Numerator is null");
    if(denominator == null)
      throw new IllegalArgumentException("Denominator is null");
    if(denominator.equals(BigInteger.ZERO))
      throw new ArithmeticException("Divide by zero.");

    //only numerator should be negative.
    if(denominator.signum() < 0)
    {
      numerator = numerator.negate();
      denominator = denominator.negate();
    }

    //create a reduced fraction
    BigInteger gcd = numerator.gcd(denominator);
    this.numerator = numerator.divide(gcd);
    this.denominator = denominator.divide(gcd);
  }

  /**
   * Constructs a BigFraction from a whole number.
   */
  public BigFraction(BigInteger numerator)
  {
    this(numerator, BigInteger.ONE, true);
  }

  public BigFraction(long numerator, long denominator)
  {
    this(BigInteger.valueOf(numerator), BigInteger.valueOf(denominator));
  }

  public BigFraction(long numerator)
  {
    this(BigInteger.valueOf(numerator), BigInteger.ONE, true);
  }

  /**
   * Constructs a BigFraction from a floating-point number.
   * 
   * Warning: round-off error in IEEE floating point numbers can result
   * in answers that are unexpected.  For example, 
   *     System.out.println(new BigFraction(1.1))
   * will print:
   *     2476979795053773/2251799813685248
   * 
   * This is because 1.1 cannot be expressed exactly in binary form.  The
   * given fraction is exactly equal to the internal representation of
   * the double-precision floating-point number.  (Which, for 1.1, is:
   * (-1)^0 * 2^0 * (1 + 0x199999999999aL / 0x10000000000000L).)
   * 
   * NOTE: In many cases, BigFraction(Double.toString(d)) may give a result
   * closer to what the user expects.
   */
  public BigFraction(double d)
  {
    if(Double.isInfinite(d))
      throw new IllegalArgumentException("double val is infinite");
    if(Double.isNaN(d))
      throw new IllegalArgumentException("double val is NaN");

    //special case - math below won't work right for 0.0 or -0.0
    if(d == 0)
    {
      numerator = BigInteger.ZERO;
      denominator = BigInteger.ONE;
      return;
    }

    final long bits = Double.doubleToLongBits(d);
    final int sign = (int)(bits >> 63) & 0x1;
    final int exponent = ((int)(bits >> 52) & 0x7ff) - 0x3ff;
    final long mantissa = bits & 0xfffffffffffffL;

    //number is (-1)^sign * 2^(exponent) * 1.mantissa
    BigInteger tmpNumerator = BigInteger.valueOf(sign==0 ? 1 : -1);
    BigInteger tmpDenominator = BigInteger.ONE;

    //use shortcut: 2^x == 1 << x.  if x is negative, shift the denominator
    if(exponent >= 0)
      tmpNumerator = tmpNumerator.multiply(BigInteger.ONE.shiftLeft(exponent));
    else
      tmpDenominator = tmpDenominator.multiply(BigInteger.ONE.shiftLeft(-exponent));

    //1.mantissa == 1 + mantissa/2^52 == (2^52 + mantissa)/2^52
    tmpDenominator = tmpDenominator.multiply(BigInteger.valueOf(0x10000000000000L));
    tmpNumerator = tmpNumerator.multiply(BigInteger.valueOf(0x10000000000000L + mantissa));

    BigInteger gcd = tmpNumerator.gcd(tmpDenominator);
    numerator = tmpNumerator.divide(gcd);
    denominator = tmpDenominator.divide(gcd);
  }

  /**
   * Constructs a BigFraction from two floating-point numbers.
   * 
   * Warning: round-off error in IEEE floating point numbers can result
   * in answers that are unexpected.  See BigFraction(double) for more
   * information.
   * 
   * NOTE: In many cases, BigFraction(Double.toString(numerator) + "/" + Double.toString(denominator))
   * may give a result closer to what the user expects.
   */
  public BigFraction(double numerator, double denominator)
  {
    if(denominator == 0)
      throw new ArithmeticException("Divide by zero.");

    BigFraction tmp = new BigFraction(numerator).divide(new BigFraction(denominator));
    this.numerator = tmp.numerator;
    this.denominator = tmp.denominator;
  }

  /**
   * Constructs a new BigFraction from the given BigDecimal object.
   */
  public BigFraction(BigDecimal d)
  {
    this(d.scale() < 0 ? d.unscaledValue().multiply(BigInteger.TEN.pow(-d.scale())) : d.unscaledValue(),
         d.scale() < 0 ? BigInteger.ONE                                             : BigInteger.TEN.pow(d.scale()));
  }

  public BigFraction(BigDecimal numerator, BigDecimal denominator)
  {
    if(denominator.equals(BigDecimal.ZERO))
      throw new ArithmeticException("Divide by zero.");

    BigFraction tmp = new BigFraction(numerator).divide(new BigFraction(denominator));
    this.numerator = tmp.numerator;
    this.denominator = tmp.denominator;
  }

  /**
   * Constructs a BigFraction from a String.  Expected format is numerator/denominator,
   * but /denominator part is optional.  Either numerator or denominator may be a floating-
   * point decimal number, which in the same format as a parameter to the
   * <code>BigDecimal(String)</code> constructor.
   * 
   * @throws NumberFormatException  if the string cannot be properly parsed.
   */
  public BigFraction(String s)
  {
    int slashPos = s.indexOf('/');
    if(slashPos < 0)
    {
      BigFraction res = new BigFraction(new BigDecimal(s));
      this.numerator = res.numerator;
      this.denominator = res.denominator;
    }
    else
    {
      BigDecimal num = new BigDecimal(s.substring(0, slashPos));
      BigDecimal den = new BigDecimal(s.substring(slashPos+1, s.length()));
      BigFraction res = new BigFraction(num, den);
      this.numerator = res.numerator;
      this.denominator = res.denominator;
    }
  }

  /**
   * Returns this + f.
   */
  public BigFraction add(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    //n1/d1 + n2/d2 = (n1*d2 + d1*n2)/(d1*d2) 
    return new BigFraction(numerator.multiply(f.denominator).add(denominator.multiply(f.numerator)),
                           denominator.multiply(f.denominator));
  }

  /**
   * Returns this + b.
   */
  public BigFraction add(BigInteger b)
  {
    if(b == null)
      throw new IllegalArgumentException("Null argument");

    //n1/d1 + n2 = (n1 + d1*n2)/d1
    return new BigFraction(numerator.add(denominator.multiply(b)),
                           denominator, true);
  }

  /**
   * Returns this + n.
   */
  public BigFraction add(long n)
  {
    return add(BigInteger.valueOf(n));
  }

  /**
   * Returns this - f.
   */
  public BigFraction subtract(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    return new BigFraction(numerator.multiply(f.denominator).subtract(denominator.multiply(f.numerator)),
                           denominator.multiply(f.denominator));
  }

  /**
   * Returns this - b.
   */
  public BigFraction subtract(BigInteger b)
  {
    if(b == null)
      throw new IllegalArgumentException("Null argument");

    return new BigFraction(numerator.subtract(denominator.multiply(b)),
                           denominator, true);
  }

  /**
   * Returns this - n.
   */
  public BigFraction subtract(long n)
  {
    return subtract(BigInteger.valueOf(n));
  }

  /**
   * Returns this * f.
   */
  public BigFraction multiply(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    return new BigFraction(numerator.multiply(f.numerator), denominator.multiply(f.denominator));
  }

  /**
   * Returns this * b.
   */
  public BigFraction multiply(BigInteger b)
  {
    if(b == null)
      throw new IllegalArgumentException("Null argument");

    return new BigFraction(numerator.multiply(b), denominator);
  }

  /**
   * Returns this * n.
   */
  public BigFraction multiply(long n)
  {
    return multiply(BigInteger.valueOf(n));
  }

  /**
   * Returns this / f.
   */
  public BigFraction divide(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    if(f.numerator.equals(BigInteger.ZERO))
      throw new ArithmeticException("Divide by zero");

    return new BigFraction(numerator.multiply(f.denominator), denominator.multiply(f.numerator));
  }

  /**
   * Returns this / b.
   */
  public BigFraction divide(BigInteger b)
  {
    if(b == null)
      throw new IllegalArgumentException("Null argument");

    if(b.equals(BigInteger.ZERO))
      throw new ArithmeticException("Divide by zero");

    return new BigFraction(numerator, denominator.multiply(b));
  }

  /**
   * Returns this / n.
   */
  public BigFraction divide(long n)
  {
    return divide(BigInteger.valueOf(n));
  }

  /**
   * Returns this^exponent.
   */
  public BigFraction pow(int exponent)
  {
    if(exponent == 0)
      return BigFraction.ONE;
    else if (exponent == 1)
      return this;
    else if (exponent < 0)
      return new BigFraction(denominator.pow(-exponent), numerator.pow(-exponent), true);
    else
      return new BigFraction(numerator.pow(exponent), denominator.pow(exponent), true);
  }

  /**
   * Returns 1/this.
   */
  public BigFraction reciprocal()
  {
    if(this.numerator.equals(BigInteger.ZERO))
      throw new ArithmeticException("Divide by zero");

    return new BigFraction(denominator, numerator, true);
  }

  /**
   * Returns the complement of this fraction, which is equal to 1 - this.
   * Useful for probabilities/statistics.

   */
  public BigFraction complement()
  {
    return new BigFraction(denominator.subtract(numerator), denominator, true);
  }

  /**
   * Returns -this.
   */
  public BigFraction negate()
  {
    return new BigFraction(numerator.negate(), denominator, true);
  }

  /**
   * Returns -1, 0, or 1, representing the sign of this fraction.
   */
  public int signum()
  {
    return numerator.signum();
  }

  /**
   * Returns the absolute value of this.
   */
  public BigFraction abs()
  {
    return (signum() < 0 ? negate() : this);
  }

  /**
   * Returns a string representation of this, in the form
   * numerator/denominator.
   */
  public String toString()
  {
    return numerator.toString() + "/" + denominator.toString();
  }

  /**
   * Returns if this object is equal to another object.
   */
  public boolean equals(Object o)
  {
    if(!(o instanceof BigFraction))
      return false;

    BigFraction f = (BigFraction)o;
    return numerator.equals(f.numerator) && denominator.equals(f.denominator);
  }

  /**
   * Returns a hash code for this object.
   */
  public int hashCode()
  {
    //using the method generated by Eclipse, but streamlined a bit..
    return (31 + numerator.hashCode())*31 + denominator.hashCode();
  }

  /**
   * Returns a negative, zero, or positive number, indicating if this object
   * is less than, equal to, or greater than f, respectively.
   */
  public int compareTo(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    //easy case: this and f have different signs
    if(signum() != f.signum())
      return signum() - f.signum();

    //next easy case: this and f have the same denominator
    if(denominator.equals(f.denominator))
      return numerator.compareTo(f.numerator);

    //not an easy case, so first make the denominators equal then compare the numerators 
    return numerator.multiply(f.denominator).compareTo(denominator.multiply(f.numerator));
  }

  /**
   * Returns the smaller of this and f.
   */
  public BigFraction min(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    return (this.compareTo(f) <= 0 ? this : f);
  }

  /**
   * Returns the maximum of this and f.
   */
  public BigFraction max(BigFraction f)
  {
    if(f == null)
      throw new IllegalArgumentException("Null argument");

    return (this.compareTo(f) >= 0 ? this : f);
  }

  /**
   * Returns a positive BigFraction, greater than or equal to zero, and less than one.
   */
  public static BigFraction random()
  {
    return new BigFraction(Math.random());
  }

  public final BigInteger getNumerator() { return numerator; }
  public final BigInteger getDenominator() { return denominator; }

  //implementation of Number class.  may cause overflow.
  public byte   byteValue()   { return (byte) Math.max(Byte.MIN_VALUE,    Math.min(Byte.MAX_VALUE,    longValue())); }
  public short  shortValue()  { return (short)Math.max(Short.MIN_VALUE,   Math.min(Short.MAX_VALUE,   longValue())); }
  public int    intValue()    { return (int)  Math.max(Integer.MIN_VALUE, Math.min(Integer.MAX_VALUE, longValue())); }
  public long   longValue()   { return Math.round(doubleValue()); }
  public float  floatValue()  { return (float)doubleValue(); }
  public double doubleValue() { return toBigDecimal(18).doubleValue(); }

  /**
   * Returns a BigDecimal representation of this fraction.  If possible, the
   * returned value will be exactly equal to the fraction.  If not, the BigDecimal
   * will have a scale large enough to hold the same number of significant figures
   * as both numerator and denominator, or the equivalent of a double-precision
   * number, whichever is more.
   */
  public BigDecimal toBigDecimal()
  {
    //Implementation note:  A fraction can be represented exactly in base-10 iff its
    //denominator is of the form 2^a * 5^b, where a and b are nonnegative integers.
    //(In other words, if there are no prime factors of the denominator except for
    //2 and 5, or if the denominator is 1).  So to determine if this denominator is
    //of this form, continually divide by 2 to get the number of 2's, and then
    //continually divide by 5 to get the number of 5's.  Afterward, if the denominator
    //is 1 then there are no other prime factors.

    //Note: number of 2's is given by the number of trailing 0 bits in the number
    int twos = denominator.getLowestSetBit();
    BigInteger tmpDen = denominator.shiftRight(twos); // x / 2^n === x >> n

    final BigInteger FIVE = BigInteger.valueOf(5);
    int fives = 0;
    BigInteger[] divMod = null;

    //while(tmpDen % 5 == 0) { fives++; tmpDen /= 5; }
    while(BigInteger.ZERO.equals((divMod = tmpDen.divideAndRemainder(FIVE))[1]))
    {
      fives++;
      tmpDen = divMod[0];
    }

    if(BigInteger.ONE.equals(tmpDen))
    {
      //This fraction will terminate in base 10, so it can be represented exactly as
      //a BigDecimal.  We would now like to make the fraction of the form
      //unscaled / 10^scale.  We know that 2^x * 5^x = 10^x, and our denominator is
      //in the form 2^twos * 5^fives.  So use max(twos, fives) as the scale, and
      //multiply the numerator and deminator by the appropriate number of 2's or 5's
      //such that the denominator is of the form 2^scale * 5^scale.  (Of course, we
      //only have to actually multiply the numerator, since all we need for the
      //BigDecimal constructor is the scale.
      BigInteger unscaled = numerator;
      int scale = Math.max(twos, fives);

      if(twos < fives)
        unscaled = unscaled.shiftLeft(fives - twos); //x * 2^n === x << n
      else if (fives < twos)
        unscaled = unscaled.multiply(FIVE.pow(twos - fives));

      return new BigDecimal(unscaled, scale);
    }

    //else: this number will repeat infinitely in base-10.  So try to figure out
    //a good number of significant digits.  Start with the number of digits required
    //to represent the numerator and denominator in base-10, which is given by
    //bitLength / log[2](10).  (bitLenth is the number of digits in base-2).
    final double LG10 = 3.321928094887362; //Precomputed ln(10)/ln(2), a.k.a. log[2](10)
    int precision = Math.max(numerator.bitLength(), denominator.bitLength());
    precision = (int)Math.ceil(precision / LG10);

    //If the precision is less than 18 digits, use 18 digits so that the number
    //will be at least as accurate as a cast to a double.  For example, with
    //the fraction 1/3, precision will be 1, giving a result of 0.3.  This is
    //quite a bit different from what a user would expect.
    if(precision < 18)
      precision = 18;

    return toBigDecimal(precision);
  }

  /**
   * Returns a BigDecimal representation of this fraction, with a given precision.
   * @param precision  the number of significant figures to be used in the result.
   */
  public BigDecimal toBigDecimal(int precision)
  {
    return new BigDecimal(numerator).divide(new BigDecimal(denominator), new MathContext(precision, RoundingMode.HALF_EVEN));
  }

  //--------------------------------------------------------------------------
  //  PRIVATE FUNCTIONS
  //--------------------------------------------------------------------------

  /**
   * Private constructor, used when you can be certain that the fraction is already in
   * lowest terms.  No check is done to reduce numerator/denominator.  A check is still
   * done to maintain a positive denominator.
   * 
   * @param throwaway  unused variable, only here to signal to the compiler that this
   *                   constructor should be used.
   */
  private BigFraction(BigInteger numerator, BigInteger denominator, boolean throwaway)
  {
    if(denominator.signum() < 0)
    {
      this.numerator = numerator.negate();
      this.denominator = denominator.negate();
    }
    else
    {
      this.numerator = numerator;
      this.denominator = denominator;
    }
  }

}
Tidur
sumber
Jika arg adalah null, lemparkan NullPointerException. Sebenarnya kode akan tetap melakukannya jadi pemeriksaan Anda (dan penggantinya dengan IllegalArgumentException (adalah kode yang tidak perlu membengkak.
cletus
24
Saya tidak setuju; jika pengguna lain menggunakan kelas ini tanpa melihat sumber saya, dan mendapatkan NullPointerException, dia akan mengira ada bug dalam kode saya . Tetapi IllegalArgumentException menunjukkan bahwa dia telah melanggar kontrak yang disiratkan oleh javadoc (meskipun saya gagal menyatakannya secara eksplisit).
Kip
1
hanya sebuah pertanyaan, apa yang salah dengan Fraksi dan Fraksi Besar dalam Matematika Commons?
Mortimer
@Mortimer: tidak yakin, saya belum pernah melihatnya
Kip
61

Sebenarnya, coba ini untuk ukuran. Ini berjalan tetapi mungkin memiliki beberapa masalah:

public class BigRational extends Number implements Comparable<BigRational>, Serializable {
    public final static BigRational ZERO = new BigRational(BigInteger.ZERO, BigInteger.ONE);
    private final static long serialVersionUID = 1099377265582986378L;

    private final BigInteger numerator, denominator;

    private BigRational(BigInteger numerator, BigInteger denominator) {
        this.numerator = numerator;
        this.denominator = denominator;
    }

    private static BigRational canonical(BigInteger numerator, BigInteger denominator, boolean checkGcd) {
        if (denominator.signum() == 0) {
            throw new IllegalArgumentException("denominator is zero");
        }
        if (numerator.signum() == 0) {
            return ZERO;
        }
        if (denominator.signum() < 0) {
            numerator = numerator.negate();
            denominator = denominator.negate();
        }
        if (checkGcd) {
            BigInteger gcd = numerator.gcd(denominator);
            if (!gcd.equals(BigInteger.ONE)) {
                numerator = numerator.divide(gcd);
                denominator = denominator.divide(gcd);
            }
        }
        return new BigRational(numerator, denominator);
    }

    public static BigRational getInstance(BigInteger numerator, BigInteger denominator) {
        return canonical(numerator, denominator, true);
    }

    public static BigRational getInstance(long numerator, long denominator) {
        return canonical(new BigInteger("" + numerator), new BigInteger("" + denominator), true);
    }

    public static BigRational getInstance(String numerator, String denominator) {
        return canonical(new BigInteger(numerator), new BigInteger(denominator), true);
    }

    public static BigRational valueOf(String s) {
        Pattern p = Pattern.compile("(-?\\d+)(?:.(\\d+)?)?0*(?:e(-?\\d+))?");
        Matcher m = p.matcher(s);
        if (!m.matches()) {
            throw new IllegalArgumentException("Unknown format '" + s + "'");
        }

        // this translates 23.123e5 to 25,123 / 1000 * 10^5 = 2,512,300 / 1 (GCD)
        String whole = m.group(1);
        String decimal = m.group(2);
        String exponent = m.group(3);
        String n = whole;

        // 23.123 => 23123
        if (decimal != null) {
            n += decimal;
        }
        BigInteger numerator = new BigInteger(n);

        // exponent is an int because BigInteger.pow() takes an int argument
        // it gets more difficult if exponent needs to be outside {-2 billion,2 billion}
        int exp = exponent == null ? 0 : Integer.valueOf(exponent);
        int decimalPlaces = decimal == null ? 0 : decimal.length();
        exp -= decimalPlaces;
        BigInteger denominator;
        if (exp < 0) {
            denominator = BigInteger.TEN.pow(-exp);
        } else {
            numerator = numerator.multiply(BigInteger.TEN.pow(exp));
            denominator = BigInteger.ONE;
        }

        // done
        return canonical(numerator, denominator, true);
    }

    // Comparable
    public int compareTo(BigRational o) {
        // note: this is a bit of cheat, relying on BigInteger.compareTo() returning
        // -1, 0 or 1.  For the more general contract of compareTo(), you'd need to do
        // more checking
        if (numerator.signum() != o.numerator.signum()) {
            return numerator.signum() - o.numerator.signum();
        } else {
            // oddly BigInteger has gcd() but no lcm()
            BigInteger i1 = numerator.multiply(o.denominator);
            BigInteger i2 = o.numerator.multiply(denominator);
            return i1.compareTo(i2); // expensive!
        }
    }

    public BigRational add(BigRational o) {
        if (o.numerator.signum() == 0) {
            return this;
        } else if (numerator.signum() == 0) {
            return o;
        } else if (denominator.equals(o.denominator)) {
            return new BigRational(numerator.add(o.numerator), denominator);
        } else {
            return canonical(numerator.multiply(o.denominator).add(o.numerator.multiply(denominator)), denominator.multiply(o.denominator), true);
        }
    }


    public BigRational multiply(BigRational o) {
        if (numerator.signum() == 0 || o.numerator.signum( )== 0) {
            return ZERO;
        } else if (numerator.equals(o.denominator)) {
            return canonical(o.numerator, denominator, true);
        } else if (o.numerator.equals(denominator)) {
            return canonical(numerator, o.denominator, true);
        } else if (numerator.negate().equals(o.denominator)) {
            return canonical(o.numerator.negate(), denominator, true);
        } else if (o.numerator.negate().equals(denominator)) {
            return canonical(numerator.negate(), o.denominator, true);
        } else {
            return canonical(numerator.multiply(o.numerator), denominator.multiply(o.denominator), true);
        }
    }

    public BigInteger getNumerator() { return numerator; }
    public BigInteger getDenominator() { return denominator; }
    public boolean isInteger() { return numerator.signum() == 0 || denominator.equals(BigInteger.ONE); }
    public BigRational negate() { return new BigRational(numerator.negate(), denominator); }
    public BigRational invert() { return canonical(denominator, numerator, false); }
    public BigRational abs() { return numerator.signum() < 0 ? negate() : this; }
    public BigRational pow(int exp) { return canonical(numerator.pow(exp), denominator.pow(exp), true); }
    public BigRational subtract(BigRational o) { return add(o.negate()); }
    public BigRational divide(BigRational o) { return multiply(o.invert()); }
    public BigRational min(BigRational o) { return compareTo(o) <= 0 ? this : o; }
    public BigRational max(BigRational o) { return compareTo(o) >= 0 ? this : o; }

    public BigDecimal toBigDecimal(int scale, RoundingMode roundingMode) {
        return isInteger() ? new BigDecimal(numerator) : new BigDecimal(numerator).divide(new BigDecimal(denominator), scale, roundingMode);
    }

    // Number
    public int intValue() { return isInteger() ? numerator.intValue() : numerator.divide(denominator).intValue(); }
    public long longValue() { return isInteger() ? numerator.longValue() : numerator.divide(denominator).longValue(); }
    public float floatValue() { return (float)doubleValue(); }
    public double doubleValue() { return isInteger() ? numerator.doubleValue() : numerator.doubleValue() / denominator.doubleValue(); }

    @Override
    public String toString() { return isInteger() ? String.format("%,d", numerator) : String.format("%,d / %,d", numerator, denominator); }

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

        BigRational that = (BigRational) o;

        if (denominator != null ? !denominator.equals(that.denominator) : that.denominator != null) return false;
        if (numerator != null ? !numerator.equals(that.numerator) : that.numerator != null) return false;

        return true;
    }

    @Override
    public int hashCode() {
        int result = numerator != null ? numerator.hashCode() : 0;
        result = 31 * result + (denominator != null ? denominator.hashCode() : 0);
        return result;
    }

    public static void main(String args[]) {
        BigRational r1 = BigRational.valueOf("3.14e4");
        BigRational r2 = BigRational.getInstance(111, 7);
        dump("r1", r1);
        dump("r2", r2);
        dump("r1 + r2", r1.add(r2));
        dump("r1 - r2", r1.subtract(r2));
        dump("r1 * r2", r1.multiply(r2));
        dump("r1 / r2", r1.divide(r2));
        dump("r2 ^ 2", r2.pow(2));
    }

    public static void dump(String name, BigRational r) {
        System.out.printf("%s = %s%n", name, r);
        System.out.printf("%s.negate() = %s%n", name, r.negate());
        System.out.printf("%s.invert() = %s%n", name, r.invert());
        System.out.printf("%s.intValue() = %,d%n", name, r.intValue());
        System.out.printf("%s.longValue() = %,d%n", name, r.longValue());
        System.out.printf("%s.floatValue() = %,f%n", name, r.floatValue());
        System.out.printf("%s.doubleValue() = %,f%n", name, r.doubleValue());
        System.out.println();
    }
}

Outputnya adalah:

r1 = 31,400
r1.negate() = -31,400
r1.invert() = 1 / 31,400
r1.intValue() = 31,400
r1.longValue() = 31,400
r1.floatValue() = 31,400.000000
r1.doubleValue() = 31,400.000000

r2 = 111 / 7
r2.negate() = -111 / 7
r2.invert() = 7 / 111
r2.intValue() = 15
r2.longValue() = 15
r2.floatValue() = 15.857142
r2.doubleValue() = 15.857143

r1 + r2 = 219,911 / 7
r1 + r2.negate() = -219,911 / 7
r1 + r2.invert() = 7 / 219,911
r1 + r2.intValue() = 31,415
r1 + r2.longValue() = 31,415
r1 + r2.floatValue() = 31,415.857422
r1 + r2.doubleValue() = 31,415.857143

r1 - r2 = 219,689 / 7
r1 - r2.negate() = -219,689 / 7
r1 - r2.invert() = 7 / 219,689
r1 - r2.intValue() = 31,384
r1 - r2.longValue() = 31,384
r1 - r2.floatValue() = 31,384.142578
r1 - r2.doubleValue() = 31,384.142857

r1 * r2 = 3,485,400 / 7
r1 * r2.negate() = -3,485,400 / 7
r1 * r2.invert() = 7 / 3,485,400
r1 * r2.intValue() = 497,914
r1 * r2.longValue() = 497,914
r1 * r2.floatValue() = 497,914.281250
r1 * r2.doubleValue() = 497,914.285714

r1 / r2 = 219,800 / 111
r1 / r2.negate() = -219,800 / 111
r1 / r2.invert() = 111 / 219,800
r1 / r2.intValue() = 1,980
r1 / r2.longValue() = 1,980
r1 / r2.floatValue() = 1,980.180176
r1 / r2.doubleValue() = 1,980.180180

r2 ^ 2 = 12,321 / 49
r2 ^ 2.negate() = -12,321 / 49
r2 ^ 2.invert() = 49 / 12,321
r2 ^ 2.intValue() = 251
r2 ^ 2.longValue() = 251
r2 ^ 2.floatValue() = 251.448975
r2 ^ 2.doubleValue() = 251.448980
cletus
sumber
30

Saya mencoba menggunakan pecahan tepat di Jawa.

Apache Commons Math telah memiliki kelas Pecahan cukup lama. Sering kali jawaban untuk, "Wah, saya berharap Java memiliki sesuatu seperti X di pustaka inti!" dapat ditemukan di bawah payung pustaka Apache Commons .

yawmark
sumber
2
Saya akan memberi tahu Anda mengapa ini sangat rendah, pustaka Apache Commons tidak ramah pemula. Pertama tidak ada tautan langsung untuk diunduh di halaman itu (tersembunyi di menu bilah sisi), kedua tidak ada instruksi tentang cara menggunakannya (menambahkan jar ke jalur pembuatan Anda), ketiga saya mendapat kesalahan classDefNotFound setelah menambahkan semuanya dengan cara apapun . Jadi Anda tidak mendapatkan suara positif dari kami orang-orang yang hanya tahu cara menyalin dan menempel.
Noumenon
@Noumenon bagaimana menggunakan manajer pembangunan (misalnya maven) dan hanya menambahkan ketergantungan di POM?
eugene.polschikov
1
Saya ingin melihat sedikit uraian "Bagaimana menggunakan ini dalam proyek Anda" untuk para noob. Saran itu bisa masuk ke sana. Meskipun demikian, saya telah menemukan cara melakukannya dan menggunakannya di aplikasi pabrik saya yang memerlukan tampilan pecahan inci, dan saya tidak pernah kembali untuk memberi Anda suara positif. Jadi terima kasih, ini terlambat.
Noumenon
Itu umpan balik yang adil. Ini ucapan terima kasih saya yang terlambat juga! :)
yawmark
Yang ini cukup mudah digunakan.
Eric Wang
24

Tolong jadikan itu tipe yang tidak bisa diubah! Nilai pecahan tidak berubah - setengah tidak menjadi sepertiga, misalnya. Alih-alih setDenominator, Anda bisa memiliki withDenominator yang mengembalikan nilai baru pecahan yang memiliki pembilang yang sama tetapi penyebut yang ditentukan.

Hidup itu jauh lebih mudah dengan tipe yang tidak bisa diubah.

Mengganti sama dengan dan kode hash akan masuk akal juga, sehingga bisa digunakan di peta dan set. Poin Outlaw Programmer tentang operator aritmatika dan pemformatan string juga bagus.

Sebagai panduan umum, lihat BigInteger dan BigDecimal. Mereka tidak melakukan hal yang sama, tetapi mereka cukup mirip untuk memberi Anda ide-ide bagus.

Jon Skeet
sumber
5
"Tolong jadikan itu tipe yang tidak bisa diubah! Nilai pecahan tidak berubah - setengah tidak menjadi sepertiga, misalnya." List / tuple / vector (1, 2, 3, 4) juga tidak menjadi nilai (4, 3, 2, 1), namun tampaknya tidak mengganggu kebanyakan orang yang membuat list berubah status. Bukannya saya tidak setuju dengan kekekalan untuk pecahan, tapi itu layak mendapat argumen yang lebih baik. Rasanya seperti sebuah nilai lebih dari sekedar sekumpulan negara. Apakah ekspektasi programmer adalah alasan yang tepat untuk dipandu? Saya tidak 100% yakin, tapi sepertinya itu ide yang bagus.
Jonas Kölker
2
Nah, dalam kehidupan nyata daftar memang berubah: bagaimana Anda menulis daftar belanja? Anda mulai dengan selembar kertas kosong, dan menulis di atasnya. Setengah jalan, Anda masih akan menyebutnya "daftar belanja". Karena itu, pemrograman fungsional berusaha keras untuk membuat daftar genap tidak dapat diubah ...
Jon Skeet
7

Yah, untuk satu, aku akan menyingkirkan penyetel dan membuat Fraksi tidak berubah.

Anda mungkin juga menginginkan metode untuk menambah, mengurangi, dll., Dan mungkin beberapa cara untuk mendapatkan representasi dalam berbagai format String.

EDIT: Saya mungkin akan menandai bidang sebagai 'final' untuk menandakan niat saya, tetapi saya rasa itu bukan masalah besar ...

Programmer Penjahat
sumber
2
Saya ingin tahu berapa banyak jawaban "buat itu tidak berubah" yang akan kita dapatkan :)
Jon Skeet
5
  • Agak tidak berguna tanpa metode aritmatika seperti add () dan multiply (), dll.
  • Anda pasti harus mengganti equals () dan hashCode ().
  • Anda harus menambahkan metode untuk menormalkan pecahan, atau melakukannya secara otomatis. Pikirkan apakah Anda ingin 1/2 dan 2/4 dianggap sama atau tidak - ini berimplikasi pada metode equals (), hashCode (), dan bandingkanTo ().
Michael Borgwardt
sumber
5

Saya perlu mengurutkan mereka dari yang terkecil ke terbesar, jadi pada akhirnya saya perlu merepresentasikannya sebagai dobel juga

Tidak terlalu penting. (Faktanya jika Anda ingin menangani persamaan dengan benar, jangan mengandalkan double untuk bekerja dengan benar.) Jika b * d positif, a / b <c / d if ad <bc. Jika ada bilangan bulat negatif yang terlibat, itu dapat ditangani dengan tepat ...

Saya mungkin menulis ulang sebagai:

public int compareTo(Fraction frac)
{
    // we are comparing this=a/b with frac=c/d 
    // by multiplying both sides by bd.
    // If bd is positive, then a/b < c/d <=> ad < bc.
    // If bd is negative, then a/b < c/d <=> ad > bc.
    // If bd is 0, then you've got other problems (either b=0 or d=0)
    int d = frac.getDenominator();
    long ad = (long)this.numerator * d;
    long bc = (long)this.denominator * frac.getNumerator();
    long diff = ((long)d*this.denominator > 0) ? (ad-bc) : (bc-ad);
    return (diff > 0 ? 1 : (diff < 0 ? -1 : 0));
}

Penggunaan di longsini adalah untuk memastikan tidak ada luapan jika Anda mengalikan dua besarint s . handle Jika Anda dapat menjamin bahwa penyebut selalu non-negatif (jika negatif, negasikan saja pembilang dan penyebutnya), maka Anda dapat menyingkirkan keharusan untuk memeriksa apakah b * d positif dan menyimpan beberapa langkah. Saya tidak yakin perilaku apa yang Anda cari dengan penyebut nol.

Tidak yakin bagaimana kinerja dibandingkan dengan penggunaan ganda untuk membandingkan. (Yaitu, jika Anda sangat peduli dengan kinerja) Berikut adalah metode pengujian yang saya gunakan untuk memeriksa. (Tampaknya berfungsi dengan baik.)

public static void main(String[] args)
{
    int a = Integer.parseInt(args[0]);
    int b = Integer.parseInt(args[1]);
    int c = Integer.parseInt(args[2]);
    int d = Integer.parseInt(args[3]);
    Fraction f1 = new Fraction(a,b); 
    Fraction f2 = new Fraction(c,d);
    int rel = f1.compareTo(f2);
    String relstr = "<=>";
    System.out.println(a+"/"+b+" "+relstr.charAt(rel+1)+" "+c+"/"+d);
}

(ps Anda mungkin mempertimbangkan restrukturisasi untuk diimplementasikan Comparableatau Comparatoruntuk kelas Anda.)

Jason S
sumber
Ini tidak benar jika, misalnya, a = 1, b = 3, c = -2, d = -3. Jika b dan d bertanda positif maka benar bahwa a / b <c / d jika dan hanya jika ad <bc.
Luke Woodward
Argh, kualifikasi saya salah. (terima kasih!) Kondisinya harus jika bd> 0.
Jason S
Benar. Lebih tepatnya, a / b <c / d <=> ac <bd benar asalkan bd> 0. Jika bd <0, sebaliknya benar. (Jika bd = 0, maka Anda memiliki pecahan pantat. :-))
Paul Brinkley
Menutup. maksud Anda a / b <c / d <=> ad <bc for bd> 0. (Saya melakukannya dengan benar pertama kali di komentar kode saya!)
Jason S
4

Satu peningkatan yang sangat kecil berpotensi untuk menyimpan nilai ganda yang Anda hitung sehingga Anda hanya menghitungnya pada akses pertama. Ini tidak akan menjadi kemenangan besar kecuali Anda sering mengakses nomor ini, tetapi itu juga tidak terlalu sulit untuk dilakukan.

Satu poin tambahan mungkin adalah pengecekan kesalahan yang Anda lakukan di penyebut ... Anda secara otomatis mengubah 0 menjadi 1. Tidak yakin apakah ini benar untuk aplikasi khusus Anda, tetapi secara umum jika seseorang mencoba membagi dengan 0, ada sesuatu yang sangat salah . Saya akan membiarkan ini melempar pengecualian (pengecualian khusus jika Anda merasa itu diperlukan) daripada mengubah nilai dengan cara yang tampaknya sewenang-wenang yang tidak diketahui pengguna.

Berkebalikan dengan beberapa komentar lain, tentang menambahkan metode untuk menambah pengurangan, dll ... karena Anda tidak menyebutkan membutuhkannya, saya berasumsi Anda tidak membutuhkannya. Dan kecuali Anda sedang membangun perpustakaan yang benar-benar akan digunakan di banyak tempat atau oleh orang lain, gunakan YAGNI (Anda tidak akan membutuhkannya, jadi seharusnya tidak ada di sana.)

Beska
sumber
Fakta bahwa dia memiliki getNumerator () dan getDenominator () membuat saya percaya dia sedang membuat pecahan baru DI LUAR kelas ini. Logika itu mungkin ada di sini jika memang ada.
Outlaw Programmer
+1 Secara diam-diam mengubah 0 menjadi 1 di penyebut adalah resep bencana.
maaartinus
4

Ada beberapa cara untuk meningkatkan ini atau jenis nilai apa pun:

  • Buat kelas Anda tidak berubah , termasuk membuat pembilang dan penyebut menjadi final
  • Secara otomatis mengonversi pecahan ke bentuk kanonik , misalnya 2/4 -> 1/2
  • Implementasikan toString ()
  • Menerapkan "public static Fraction valueOf (String s)" untuk mengkonversi dari string ke pecahan. Terapkan metode pabrik serupa untuk mengonversi dari int, double, dll.
  • Menerapkan penjumlahan, perkalian, dll
  • Tambahkan konstruktor dari bilangan bulat
  • Timpa sama dengan / kode hash
  • Pertimbangkan untuk menjadikan Fraction sebagai antarmuka dengan implementasi yang beralih ke BigInteger jika diperlukan
  • Pertimbangkan Nomor sub-klasifikasi
  • Pertimbangkan untuk menyertakan konstanta bernama untuk nilai umum seperti 0 dan 1
  • Pertimbangkan untuk membuatnya dapat berseri
  • Ujilah pembagian dengan nol
  • Dokumentasikan API Anda

Pada dasarnya, lihat API untuk kelas nilai lainnya seperti Double , Integer dan lakukan apa yang mereka lakukan :)

Dave Ray
sumber
3

Jika Anda mengalikan pembilang dan penyebut dari satu pecahan dengan penyebut yang lain dan sebaliknya, Anda akan mendapatkan dua pecahan (yang nilainya masih sama) dengan penyebut yang sama dan Anda dapat membandingkan pembilangnya secara langsung. Oleh karena itu, Anda tidak perlu menghitung nilai ganda:

public int compareTo(Fraction frac) {
    int t = this.numerator * frac.getDenominator();
    int f = frac.getNumerator() * this.denominator;
    if(t>f) return 1;
    if(f>t) return -1;
    return 0;
}
Francisco Canedo
sumber
Gagal jika frac.getDenominator () dan this.denominator memiliki tanda yang berlawanan. (lihat posting saya.) Juga Anda harus berhati-hati terhadap fakta bahwa penggandaan dapat melimpah.
Jason S
Ah ya itu benar. Tapi dalam kasus itu saya lebih suka implementasi Kip, yang setidaknya bisa saya pahami. ;)
Francisco Canedo
Saya akan menunjukkan bahwa dalam implementasi saya, hanya pembilangnya yang bisa negatif. Saya juga menggunakan BigIntegers sehingga tidak akan pernah ada overflow (dengan mengorbankan beberapa kinerja, tentu saja).
Kip
2

bagaimana saya akan meningkatkan kode itu:

  1. konstruktor berdasarkan Fraksi String (String s) // harapkan "angka / angka"
  2. salinan konstruktor Fraksi (salinan pecahan)
  3. mengganti metode klon
  4. mengimplementasikan metode equals, toString dan hashcode
  5. mengimplementasikan antarmuka java.io.Serializable, Comparable
  6. metode "double getDoubleValue ()"
  7. metode tambahkan / bagi / dll ...
  8. Saya akan menjadikan kelas itu sebagai tidak dapat diubah (tidak ada penyetel)
Pierre
sumber
Daftar yang cukup bagus. Mungkin tidak ada kebutuhan untuk klon / serializable tetapi yang lainnya masuk akal.
Outlaw Programmer
@OutlawProgrammer: Ya, baik 8 atau 3. Cloneable immutable adalah tidak masuk akal.
maaartinus
2

Anda sudah memiliki fungsi CompareTo ... Saya akan mengimplementasikan antarmuka Comparable.

Mungkin tidak terlalu penting untuk apa pun yang akan Anda lakukan dengannya.

Dave Costa
sumber
2

Jika Anda suka berpetualang, lihat JScience . Ini memiliki Rationalkelas yang mewakili pecahan.

Zach Scrivena
sumber
2

Secara khusus : Apakah ada cara yang lebih baik untuk menangani penyebutan nol? Menyetel penyebut ke 1 terasa sangat sewenang-wenang. Bagaimana saya bisa melakukan ini dengan benar?

Saya akan mengatakan melempar ArithmeticException untuk membagi dengan nol, karena itulah yang sebenarnya terjadi:

public Fraction(int numerator, int denominator) {
    if(denominator == 0)
        throw new ArithmeticException("Divide by zero.");
    this.numerator = numerator;
    this.denominator = denominator;
}

Alih-alih "Bagi dengan nol", Anda mungkin ingin membuat pesan yang mengatakan "Bagi dengan nol: Penyebut untuk Pecahan adalah nol."

Tidur
sumber
1

Setelah Anda membuat objek pecahan, mengapa Anda ingin mengizinkan objek lain menyetel pembilang atau penyebutnya? Saya pikir ini harus dibaca saja. Itu membuat objek itu tidak berubah ...

Juga ... mengatur penyebut ke nol harus membuang pengecualian argumen yang tidak valid (saya tidak tahu apa itu di Jawa)

Jason Punyon
sumber
Atau lempar ArithmeticException baru ("Bagi dengan nol.")
Kip
1

Timothy Budd memiliki implementasi kelas Rasional yang bagus dalam "Struktur Data di C ++". Bahasa berbeda, tentu saja, tetapi port ke Java dengan sangat baik.

Saya akan merekomendasikan lebih banyak konstruktor. Konstruktor default akan memiliki pembilang 0, penyebut 1. Konstruktor arg tunggal akan mengasumsikan penyebut 1. Pikirkan bagaimana pengguna Anda dapat menggunakan kelas ini.

Tidak ada cek untuk penyebut nol? Pemrograman berdasarkan kontrak akan membuat Anda menambahkannya.

duffymo
sumber
1

Saya akan merekomendasikan ketiga atau kelima atau apapun untuk membuat pecahan Anda tidak berubah. Saya juga merekomendasikan agar Anda memperpanjang kelas Number . Saya mungkin akan melihat kelas Double , karena Anda mungkin ingin menerapkan banyak metode yang sama.

Anda mungkin juga harus mengimplementasikan Comparable dan Serializable karena perilaku ini mungkin diharapkan. Jadi, Anda perlu mengimplementasikan bandingkanTo (). Anda juga perlu mengganti equals () dan saya tidak bisa cukup menekankan bahwa Anda juga mengganti hashCode (). Ini mungkin salah satu dari sedikit kasus meskipun Anda tidak ingin bandingkanTo () dan sama dengan () menjadi konsisten karena pecahan dapat direduksi satu sama lain belum tentu sama.

James
sumber
1

Praktik pembersihan yang saya suka adalah hanya memiliki satu pengembalian.

 public int compareTo(Fraction frac) {
        int result = 0
        double t = this.doubleValue();
        double f = frac.doubleValue();
        if(t>f) 
           result = 1;
        else if(f>t) 
           result -1;
        return result;
    }
Milhous
sumber
1

Gunakan kelas Rasional dari perpustakaan JScience . Itu hal terbaik untuk aritmatika pecahan yang saya lihat di Jawa.

Alexander Temerev
sumber
1

Saya membersihkan jawaban cletus :

  • Menambahkan Javadoc untuk semua metode.
  • Menambahkan pemeriksaan untuk prasyarat metode.
  • Mengganti penguraian khusus valueOf(String)dengan BigInteger(String)yang lebih fleksibel dan lebih cepat.
import com.google.common.base.Splitter;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.math.RoundingMode;
import java.util.List;
import java.util.Objects;
import org.bitbucket.cowwoc.preconditions.Preconditions;

/**
 * A rational fraction, represented by {@code numerator / denominator}.
 * <p>
 * This implementation is based on <a
 * href="https://stackoverflow.com/a/474577/14731">https://stackoverflow.com/a/474577/14731</a>
 * <p>
 * @author Gili Tzabari
 */
public final class BigRational extends Number implements Comparable<BigRational>
{
    private static final long serialVersionUID = 0L;
    public static final BigRational ZERO = new BigRational(BigInteger.ZERO, BigInteger.ONE);
    public static final BigRational ONE = new BigRational(BigInteger.ONE, BigInteger.ONE);

    /**
     * Ensures the fraction the denominator is positive and optionally divides the numerator and
     * denominator by the greatest common factor.
     * <p>
     * @param numerator   a numerator
     * @param denominator a denominator
     * @param checkGcd    true if the numerator and denominator should be divided by the greatest
     *                    common factor
     * @return the canonical representation of the rational fraction
     */
    private static BigRational canonical(BigInteger numerator, BigInteger denominator,
        boolean checkGcd)
    {
        assert (numerator != null);
        assert (denominator != null);
        if (denominator.signum() == 0)
            throw new IllegalArgumentException("denominator is zero");
        if (numerator.signum() == 0)
            return ZERO;
        BigInteger newNumerator = numerator;
        BigInteger newDenominator = denominator;
        if (newDenominator.signum() < 0)
        {
            newNumerator = newNumerator.negate();
            newDenominator = newDenominator.negate();
        }
        if (checkGcd)
        {
            BigInteger gcd = newNumerator.gcd(newDenominator);
            if (!gcd.equals(BigInteger.ONE))
            {
                newNumerator = newNumerator.divide(gcd);
                newDenominator = newDenominator.divide(gcd);
            }
        }
        return new BigRational(newNumerator, newDenominator);
    }

    /**
     * @param numerator   a numerator
     * @param denominator a denominator
     * @return a BigRational having value {@code numerator / denominator}
     * @throws NullPointerException if numerator or denominator are null
     */
    public static BigRational valueOf(BigInteger numerator, BigInteger denominator)
    {
        Preconditions.requireThat(numerator, "numerator").isNotNull();
        Preconditions.requireThat(denominator, "denominator").isNotNull();
        return canonical(numerator, denominator, true);
    }

    /**
     * @param numerator   a numerator
     * @param denominator a denominator
     * @return a BigRational having value {@code numerator / denominator}
     */
    public static BigRational valueOf(long numerator, long denominator)
    {
        BigInteger bigNumerator = BigInteger.valueOf(numerator);
        BigInteger bigDenominator = BigInteger.valueOf(denominator);
        return canonical(bigNumerator, bigDenominator, true);
    }

    /**
     * @param value the parameter value
     * @param name  the parameter name
     * @return the BigInteger representation of the parameter
     * @throws NumberFormatException if value is not a valid representation of BigInteger
     */
    private static BigInteger requireBigInteger(String value, String name)
        throws NumberFormatException
    {
        try
        {
            return new BigInteger(value);
        }
        catch (NumberFormatException e)
        {
            throw (NumberFormatException) new NumberFormatException("Invalid " + name + ": " + value).
                initCause(e);
        }
    }

    /**
     * @param numerator   a numerator
     * @param denominator a denominator
     * @return a BigRational having value {@code numerator / denominator}
     * @throws NullPointerException     if numerator or denominator are null
     * @throws IllegalArgumentException if numerator or denominator are empty
     * @throws NumberFormatException    if numerator or denominator are not a valid representation of
     *                                  BigDecimal
     */
    public static BigRational valueOf(String numerator, String denominator)
        throws NullPointerException, IllegalArgumentException, NumberFormatException
    {
        Preconditions.requireThat(numerator, "numerator").isNotNull().isNotEmpty();
        Preconditions.requireThat(denominator, "denominator").isNotNull().isNotEmpty();
        BigInteger bigNumerator = requireBigInteger(numerator, "numerator");
        BigInteger bigDenominator = requireBigInteger(denominator, "denominator");
        return canonical(bigNumerator, bigDenominator, true);
    }

    /**
     * @param value a string representation of a rational fraction (e.g. "12.34e5" or "3/4")
     * @return a BigRational representation of the String
     * @throws NullPointerException     if value is null
     * @throws IllegalArgumentException if value is empty
     * @throws NumberFormatException    if numerator or denominator are not a valid representation of
     *                                  BigDecimal
     */
    public static BigRational valueOf(String value)
        throws NullPointerException, IllegalArgumentException, NumberFormatException
    {
        Preconditions.requireThat(value, "value").isNotNull().isNotEmpty();
        List<String> fractionParts = Splitter.on('/').splitToList(value);
        if (fractionParts.size() == 1)
            return valueOfRational(value);
        if (fractionParts.size() == 2)
            return BigRational.valueOf(fractionParts.get(0), fractionParts.get(1));
        throw new IllegalArgumentException("Too many slashes: " + value);
    }

    /**
     * @param value a string representation of a rational fraction (e.g. "12.34e5")
     * @return a BigRational representation of the String
     * @throws NullPointerException     if value is null
     * @throws IllegalArgumentException if value is empty
     * @throws NumberFormatException    if numerator or denominator are not a valid representation of
     *                                  BigDecimal
     */
    private static BigRational valueOfRational(String value)
        throws NullPointerException, IllegalArgumentException, NumberFormatException
    {
        Preconditions.requireThat(value, "value").isNotNull().isNotEmpty();
        BigDecimal bigDecimal = new BigDecimal(value);
        int scale = bigDecimal.scale();
        BigInteger numerator = bigDecimal.unscaledValue();
        BigInteger denominator;
        if (scale > 0)
            denominator = BigInteger.TEN.pow(scale);
        else
        {
            numerator = numerator.multiply(BigInteger.TEN.pow(-scale));
            denominator = BigInteger.ONE;
        }

        return canonical(numerator, denominator, true);
    }

    private final BigInteger numerator;
    private final BigInteger denominator;

    /**
     * @param numerator   the numerator
     * @param denominator the denominator
     * @throws NullPointerException if numerator or denominator are null
     */
    private BigRational(BigInteger numerator, BigInteger denominator)
    {
        Preconditions.requireThat(numerator, "numerator").isNotNull();
        Preconditions.requireThat(denominator, "denominator").isNotNull();
        this.numerator = numerator;
        this.denominator = denominator;
    }

    /**
     * @return the numerator
     */
    public BigInteger getNumerator()
    {
        return numerator;
    }

    /**
     * @return the denominator
     */
    public BigInteger getDenominator()
    {
        return denominator;
    }

    @Override
    @SuppressWarnings("AccessingNonPublicFieldOfAnotherObject")
    public int compareTo(BigRational other)
    {
        Preconditions.requireThat(other, "other").isNotNull();

        // canonical() ensures denominator is positive
        if (numerator.signum() != other.numerator.signum())
            return numerator.signum() - other.numerator.signum();

        // Set the denominator to a common multiple before comparing the numerators
        BigInteger first = numerator.multiply(other.denominator);
        BigInteger second = other.numerator.multiply(denominator);
        return first.compareTo(second);
    }

    /**
     * @param other another rational fraction
     * @return the result of adding this object to {@code other}
     * @throws NullPointerException if other is null
     */
    @SuppressWarnings("AccessingNonPublicFieldOfAnotherObject")
    public BigRational add(BigRational other)
    {
        Preconditions.requireThat(other, "other").isNotNull();
        if (other.numerator.signum() == 0)
            return this;
        if (numerator.signum() == 0)
            return other;
        if (denominator.equals(other.denominator))
            return new BigRational(numerator.add(other.numerator), denominator);
        return canonical(numerator.multiply(other.denominator).
            add(other.numerator.multiply(denominator)),
            denominator.multiply(other.denominator), true);
    }

    /**
     * @param other another rational fraction
     * @return the result of subtracting {@code other} from this object
     * @throws NullPointerException if other is null
     */
    @SuppressWarnings("AccessingNonPublicFieldOfAnotherObject")
    public BigRational subtract(BigRational other)
    {
        return add(other.negate());
    }

    /**
     * @param other another rational fraction
     * @return the result of multiplying this object by {@code other}
     * @throws NullPointerException if other is null
     */
    @SuppressWarnings("AccessingNonPublicFieldOfAnotherObject")
    public BigRational multiply(BigRational other)
    {
        Preconditions.requireThat(other, "other").isNotNull();
        if (numerator.signum() == 0 || other.numerator.signum() == 0)
            return ZERO;
        if (numerator.equals(other.denominator))
            return canonical(other.numerator, denominator, true);
        if (other.numerator.equals(denominator))
            return canonical(numerator, other.denominator, true);
        if (numerator.negate().equals(other.denominator))
            return canonical(other.numerator.negate(), denominator, true);
        if (other.numerator.negate().equals(denominator))
            return canonical(numerator.negate(), other.denominator, true);
        return canonical(numerator.multiply(other.numerator), denominator.multiply(other.denominator),
            true);
    }

    /**
     * @param other another rational fraction
     * @return the result of dividing this object by {@code other}
     * @throws NullPointerException if other is null
     */
    public BigRational divide(BigRational other)
    {
        return multiply(other.invert());
    }

    /**
     * @return true if the object is a whole number
     */
    public boolean isInteger()
    {
        return numerator.signum() == 0 || denominator.equals(BigInteger.ONE);
    }

    /**
     * Returns a BigRational whose value is (-this).
     * <p>
     * @return -this
     */
    public BigRational negate()
    {
        return new BigRational(numerator.negate(), denominator);
    }

    /**
     * @return a rational fraction with the numerator and denominator swapped
     */
    public BigRational invert()
    {
        return canonical(denominator, numerator, false);
    }

    /**
     * @return the absolute value of this {@code BigRational}
     */
    public BigRational abs()
    {
        if (numerator.signum() < 0)
            return negate();
        return this;
    }

    /**
     * @param exponent exponent to which both numerator and denominator is to be raised.
     * @return a BigRational whose value is (this<sup>exponent</sup>).
     */
    public BigRational pow(int exponent)
    {
        return canonical(numerator.pow(exponent), denominator.pow(exponent), true);
    }

    /**
     * @param other another rational fraction
     * @return the minimum of this object and the other fraction
     */
    public BigRational min(BigRational other)
    {
        if (compareTo(other) <= 0)
            return this;
        return other;
    }

    /**
     * @param other another rational fraction
     * @return the maximum of this object and the other fraction
     */
    public BigRational max(BigRational other)
    {
        if (compareTo(other) >= 0)
            return this;
        return other;
    }

    /**
     * @param scale        scale of the BigDecimal quotient to be returned
     * @param roundingMode the rounding mode to apply
     * @return a BigDecimal representation of this object
     * @throws NullPointerException if roundingMode is null
     */
    public BigDecimal toBigDecimal(int scale, RoundingMode roundingMode)
    {
        Preconditions.requireThat(roundingMode, "roundingMode").isNotNull();
        if (isInteger())
            return new BigDecimal(numerator);
        return new BigDecimal(numerator).divide(new BigDecimal(denominator), scale, roundingMode);
    }

    @Override
    public int intValue()
    {
        return (int) longValue();
    }

    @Override
    public long longValue()
    {
        if (isInteger())
            return numerator.longValue();
        return numerator.divide(denominator).longValue();
    }

    @Override
    public float floatValue()
    {
        return (float) doubleValue();
    }

    @Override
    public double doubleValue()
    {
        if (isInteger())
            return numerator.doubleValue();
        return numerator.doubleValue() / denominator.doubleValue();
    }

    @Override
    @SuppressWarnings("AccessingNonPublicFieldOfAnotherObject")
    public boolean equals(Object o)
    {
        if (this == o)
            return true;
        if (!(o instanceof BigRational))
            return false;
        BigRational other = (BigRational) o;

        return numerator.equals(other.denominator) && Objects.equals(denominator, other.denominator);
    }

    @Override
    public int hashCode()
    {
        return Objects.hash(numerator, denominator);
    }

    /**
     * Returns the String representation: {@code numerator / denominator}.
     */
    @Override
    public String toString()
    {
        if (isInteger())
            return String.format("%,d", numerator);
        return String.format("%,d / %,d", numerator, denominator);
    }
}
Gili
sumber
0

Komentar awal:

Jangan pernah menulis ini:

if ( condition ) statement;

Ini jauh lebih baik

if ( condition ) { statement };

Ciptakan saja untuk menciptakan kebiasaan yang baik.

Dengan membuat kelas tidak dapat diubah seperti yang disarankan, Anda juga dapat memanfaatkan fungsi ganda untuk melakukan operasi equals dan hashCode serta bandingkanTo

Ini versi cepat kotor saya:

public final class Fraction implements Comparable {

    private final int numerator;
    private final int denominator;
    private final Double internal;

    public static Fraction createFraction( int numerator, int denominator ) { 
        return new Fraction( numerator, denominator );
    }

    private Fraction(int numerator, int denominator) {
        this.numerator   = numerator;
        this.denominator = denominator;
        this.internal = ((double) numerator)/((double) denominator);
    }


    public int getNumerator() {
        return this.numerator;
    }

    public int getDenominator() {
        return this.denominator;
    }


    private double doubleValue() {
        return internal;
    }

    public int compareTo( Object o ) {
        if ( o instanceof Fraction ) { 
            return internal.compareTo( ((Fraction)o).internal );
        }
        return 1;
    }

    public boolean equals( Object o ) {
          if ( o instanceof Fraction ) {  
             return this.internal.equals( ((Fraction)o).internal );
          } 
          return false;
    }

    public int hashCode() { 
        return internal.hashCode();
    }



    public String toString() { 
        return String.format("%d/%d", numerator, denominator );
    }

    public static void main( String [] args ) { 
        System.out.println( Fraction.createFraction( 1 , 2 ) ) ;
        System.out.println( Fraction.createFraction( 1 , 2 ).hashCode() ) ;
        System.out.println( Fraction.createFraction( 1 , 2 ).compareTo( Fraction.createFraction(2,4) ) ) ;
        System.out.println( Fraction.createFraction( 1 , 2 ).equals( Fraction.createFraction(4,8) ) ) ;
        System.out.println( Fraction.createFraction( 3 , 9 ).equals( Fraction.createFraction(1,3) ) ) ;
    }       

}

Tentang metode pabrik statis, mungkin berguna nanti, jika Anda membuat subkelas Fraksi untuk menangani hal-hal yang lebih kompleks, atau jika Anda memutuskan untuk menggunakan kumpulan untuk objek yang paling sering digunakan.

Mungkin tidak demikian, saya hanya ingin menunjukkannya. :)

Lihat item pertama Java Efektif .

OscarRyz
sumber
0

Mungkin berguna untuk menambahkan hal-hal sederhana seperti membalas, mendapatkan sisa, dan utuh.

Darth Joshua
sumber
jawaban ini cocok sebagai komentar.
Jasonw
Sangat menyesal atas balasan yang terlambat tetapi saya yakin ada jumlah perwakilan minimum (50?) Yang diperlukan untuk mengomentari jawaban yang tidak saya miliki ...
Darth Joshua
0

Meskipun Anda memiliki metode bandingkanTo (), jika Anda ingin menggunakan utilitas seperti Collections.sort (), Anda juga harus mengimplementasikan Comparable.

public class Fraction extends Number implements Comparable<Fraction> {
 ...
}

Selain itu, untuk tampilan yang cantik, saya sarankan untuk mengganti toString ()

public String toString() {
    return this.getNumerator() + "/" + this.getDenominator();
}

Dan akhirnya, saya akan membuat kelas menjadi publik sehingga Anda dapat menggunakannya dari paket yang berbeda.

Kenny Cason
sumber
0

Fungsi ini menyederhanakan dengan menggunakan algoritma eucledian cukup berguna saat mendefinisikan pecahan

 public Fraction simplify(){


     int safe;
     int h= Math.max(numerator, denominator);
     int h2 = Math.min(denominator, numerator);

     if (h == 0){

         return new Fraction(1,1);
     }

     while (h>h2 && h2>0){

          h = h - h2;
          if (h>h2){

              safe = h;
              h = h2;
              h2 = safe;

          }  

     }

  return new Fraction(numerator/h,denominator/h);

 }
Brennan
sumber
0

Untuk implementasi Fraksi / Rasional kelas industri, saya akan menerapkannya sehingga dapat mewakili NaN, infinity positif, infinity negatif, dan nol negatif opsional dengan semantik operasional persis sama dengan standar IEEE 754 untuk aritmatika floating point (ini juga memudahkan konversi ke / dari nilai floating point). Plus, karena perbandingan dengan nol, satu, dan nilai khusus di atas hanya perlu sederhana, tetapi perbandingan gabungan pembilang dan penyebut terhadap 0 dan 1 - saya akan menambahkan beberapa metode isXXX dan bandingkanToXXX untuk kemudahan penggunaan (mis. Eq0 () akan gunakan pembilang == 0 && penyebut! = 0 di belakang layar alih-alih membiarkan klien membandingkan dengan contoh bernilai nol). Beberapa nilai yang telah ditentukan secara statis (ZERO, ONE, TWO, TEN, ONE_TENTH, NAN, etc.) juga berguna, karena mereka muncul di beberapa tempat sebagai nilai konstan. Ini adalah IMHO cara terbaik.

Tiamin
sumber
0

Fraksi Kelas:

     public class Fraction {
        private int num;            // numerator 
        private int denom;          // denominator 
        // default constructor
        public Fraction() {}
        // constructor
        public Fraction( int a, int b ) {
            num = a;
            if ( b == 0 )
                throw new ZeroDenomException();
            else
                denom = b;
        }
        // return string representation of ComplexNumber
        @Override
        public String toString() {
            return "( " + num + " / " + denom + " )";
        }
        // the addition operation
        public Fraction add(Fraction x){
            return new Fraction(
                    x.num * denom + x.denom * num, x.denom * denom );
        }
        // the multiplication operation
        public Fraction multiply(Fraction x) {
            return new Fraction(x.num * num, x.denom * denom);
        } 
}

Program utama:

    static void main(String[] args){
    Scanner input = new Scanner(System.in);
    System.out.println("Enter numerator and denominator of first fraction");
    int num1 =input.nextInt();
    int denom1 =input.nextInt();
    Fraction x = new Fraction(num1, denom1);
    System.out.println("Enter numerator and denominator of second fraction");
    int num2 =input.nextInt();
    int denom2 =input.nextInt();
    Fraction y = new Fraction(num2, denom2);
    Fraction result = new Fraction();
    System.out.println("Enter required operation: A (Add), M (Multiply)");
    char op = input.next().charAt(0);
    if(op == 'A') {
        result = x.add(y);
        System.out.println(x + " + " + y + " = " + result);
    }
BasmaSH
sumber