Skip to content

Algorithms to find the best rational approximation to a given floating-point number and to find the gcd and lcm of a given list of floating-point numbers

Notifications You must be signed in to change notification settings

alidasdan/best-rational-approximation

Repository files navigation

best-rational-approximation

This package contains algorithms to find the best rational approximation to a given floating-point number and to find the gcd and lcm of a given list of floating-point numbers.

BEST RATIONAL APPROXIMATION PROBLEM

A fraction n/d, where d is at most a given limit l, is the best rational approximation to a target real number t if the absolute difference (the absolute error) between n/d and t is the smallest among all the fractions each of whose denominator is at most l.

Since this is a well-known problem in mathematics, I will not give more details. The best sources are already available on the Web.

ALGORITHMS FOR BEST RATIONAL APPROXIMATION

I give the implementations of three algorithms for best rational approximation in three different Python programs:

  • 'ad_rat_by_cont_frac.py': Algorithm using continued fractions.
  • 'ad_rat_by_farey.py': Algorithm using the Farey sequence.
  • 'ad_rat_by_fast_farey.py': Faster algorithm using the Farey sequence.

The algorithm in 'ad_rat_by_cont_frac.py' is not difficult to design from scratch but my implementation is based closely on an implementation in [6].

I designed the other two algorithms: The algorithm using the Farey sequence is simple but is definitely not trivial. For example, the implementation at [2], which is one of the top links from a google search on 'best rational approximation', does not work correctly in all cases, e.g., it fails to find the best rational approximation n/d to pi when d is upper-bounded by 100.

The faster algorithm using the Farey sequence reduces the number of iterations drastically (except when running for the golden ratio as the target) but all algorithms work fast enough for denominator limits up to one million and potentially even beyond.

I was able to find two other references for a faster algorithm using the Farey sequence: [3] and [7]. I have not implemented the algorithm in [7] so let us focus on the algorithm in [3]. There are two implementations of this algorithm at [4] and [5]. These implementations take in two inputs: the number to approximate and a relative error threshold. My somewhat limited testing showed that they do produce the same output, which is as expected as [5] is a reimplementation of [4].

Unfortunately, as in [2], [3-5] miss many of the best rational approximations when I ran them for pi. For example, a run of either [4] or [5] for pi=3.14159265358979323844 misses 311/99 as the following output shows, where the last line is the final answer with intermediate approximations along the way

3/1	epsilon = 5.000000e-02
22/7	epsilon = 4.000000e-04
355/113	epsilon = 8.491368e-08
v=3.14159 n/d=355/113 err=0.0001

Changing the error calculation to the absolute error or removing the error rounding (see the code to understand what I mean by 'error rounding') did not change the result. As a result, I have decided not to post my implementation of the algorithm in [3] since my 'fast Farey algorithm' already produces the correct output.

One caveat to note is that real numbers are represented in computers as floating-point numbers. Beyond the binary representation issues, floating-point numbers are basically rational numbers. In other words, any implementation of these or similar algorithms actually finds the best rational approximation (i.e., as a rational number with a bounded denominator) to a given rational number. As an example, 'bc' returns 3.14159265358979323844 as pi (computed as '4*a(1)'). As a rational number, we can view this approximation as 314159265358979323844/1e+20 or approximate it to 22/7, which is the best rational approximation to pi when the denominator is limited by 10.

HOW TO RUN

Each of the three programs take the same arguments and return the output in the same format. For 'ad_rat_by_farey.py', the usage is shown below.

ad_rat_by_farey.py 
   -h/--help 
  [-e/--error=float>=0] 
   -l/--limit=int=>1 
   -t/--target=float or quoted math expr returning float

To find the best rational approximation n/d to a target floating-point number t, these programs always need a limit on d. In addition, an upper bound on the absolute error, an optional parameter, can also be input.

Here is how the output from a typical run looks like:

> ad_rat_by_farey.py -l 10 -t pi
target= 3.141593 best_rat= 22 / 7 max_denom= 10 err= -0.00126449 abs_err= 0.00126449 niter= 8

Here we run 'ad_ray_by_farey.py' to find the best rational approximation n/d to pi where d is at most 10. The program returns 22/7 as the best approximation in 8 iterations (of a loop that the algorithm contains). The absolute error is close to 1e-3.

A potentially interesting part of these programs is that these programs can also accept a math expression that returns a float. The expression must be enclosed in single or double quotes. For example, here is how to find the best rational approximation to the golden ratio:

> ad_rat_by_farey.py -l 10 -t '(1+sqrt(5))/2'
target= 1.618034 best_rat= 13 / 8 max_denom= 10 err= -0.00696601 abs_err= 0.00696601 niter= 5

This output states that the best rational approximation to the golden ratio is 13/8 with an absolute error close to 1e-2.

These programs run very fast even for very large limits.

HOW TO TEST

Type 'utest.sh'. For double-checking the approximations to pi, check out http://oeis.org/A063674 and http://oeis.org/A063673 (which respectively give the numerators and denominators of the best rational approximations to pi).

You can check the results by running an exhaustive search using 'ad_rat_by_exhaustive.py', which runs very slow for limits larger than 1000.

HOW TO COMPUTE GCD AND LCM OF A LIST OF FLOATING-POINT NUMBERS

A related problem to the best rational approximation problem is computing the greatest common divisor (gcd) and least common multiple (lcm) of a given list of floating-point numbers (or real numbers). A solution to this problem can be given using the best rational approximation algorithms. The solution has two main steps:

  • (1) finding the best rational approximation to each number in the input list;
  • (2) computing the gcd and lcm of the resulting rational numbers. Note that the gcd and lcm of rational numbers are well-defined (as in the case for integers).

The program 'ad_gcd_lcm.py' solves this problem. Without a proof, I cannot claim that this is the best solution to this problem but it works very well for all the cases I could consider and also find on the Web.

Here is how to run 'ad_gcd_lcm.py':

ad_gcd_lcm.py 
   -l/--limit=int>=1 
   -n/--nums=quoted list of at least 2 numbers or math expression returning float (w/o spaces)

This program takes in a limit on the largest denominator of any best rational approximation to any of the numbers in the list.

Here is a run with a verbose output:

> ad_gcd_lcm.py -l 10 -n 'pi 2*pi 3*pi'
input: ['pi', '2*pi', '3*pi']
input evaluated: [3.1415926535897931, 6.2831853071795862, 9.4247779607693793]
best_rats= [22/7, 44/7, 66/7]
gcd: rat= 22 / 7 val= 3.14286
lcm: rat= 132 / 7 val= 18.8571 lcm/gcd= 6
nums_div_gcd=  [1, 2, 3] max_err= 0.00120701
lcm_div_nums=  [6, 3, 2] max_err= 0.002415

We run 'ad_gcd_lcm.py' for the first three multiples of pi. The expected result is that the gcd should be equal to pi and the lcm should be equal to 6 times pi. The program correctly outputs these expected values.

'utests2.sh' tests this program.

BEST RATIONAL APPROXIMATIONS FOR THE GOLDEN RATIO AND ITS CONJUGATE

Using the Fibonacci sequence, it is trivial to come up with the best rational approximations to the golden ratio and its conjugate.

Recall that the Fibonacci sequence goes

0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, ...

where every number (except the first two) is a sum of the previous two numbers in the sequence. Also recall that the golden ratio (G) and its conjugate (C) are defined as

G = (sqrt(5) + 1) / 2 and C = (sqrt(5) - 1) / 2 (or C = 1 / G = G - 1).

Suppose F1, F2, and F3 are three consecutive Fibonnacci numbers such that F2 is less than or equal to L. Then, the observation (well known [8]) is that the best rational approximation to G whose denominator is at most L is equal to F3 / F2 and the one for C is equal to F1 / F2. For example,

L G C
2 3 / 2 1 / 2
5 8 / 5 3 / 5
10 13 / 8 5 / 8
20 21 / 13 8 / 13
30 34 / 21 13 / 21
40 55 / 34 21 / 34

If you run the programs in this repo, the results will be as shown in this table. Note that this observation also provides an easy way to test the results of such programs because the Fibonacci sequence can be computed easily.

REFERENCE

[1] R.L. Graham, D.E. Knuth, and O. Patashnik, Concrete Mathematics, 2nd Edition, Addison-Wesley Longman, 1994.

[2] John D. Cook, Code for Best Rational Approximation, URL= "http://www.johndcook.com/blog/2010/10/20/best-rational-approximation/"

[3] J. Spanier and K.B. Oldham, An Atlas of Functions, Springer-Verlag, 1987, pp. 665-7.

[4] R. J. Craig, Code for Best Rational Approximation, URL= "http://www.netlib.org/c/frac". Implements the algorithm proposed in [3] (with lots of GOTOs).

[5] Raevnos, Code for Best Rational Approximation, URL= "http://pennmush.sourcearchive.com/documentation/1.8.2p8/funmath_8c_6d20fe500d5000139804cfb8e6377087.html". Converts the code in [4] in a structured manner without GOTOs.

[6] D. Eppstein, Code for Best Rational Approximation, URL="https://www.ics.uci.edu/~eppstein/numth/frap.c". Implements the solution using continued fractions.

[7] M. Forisek, Approximating Rational Numbers by Fractions, in Fun with Algorithms: LNCS, Springer-Verlag, v. 4475, pp. 156-165, 2007.

[8] E. W. Weisstein, Golden Ratio, From MathWorld--A Wolfram Web Resource. URL="http://mathworld.wolfram.com/GoldenRatio.html".

About

Algorithms to find the best rational approximation to a given floating-point number and to find the gcd and lcm of a given list of floating-point numbers

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published