fractional
NOTE: If the mathematical inserts in this vignette are not displaying correctly or even replaced by messages [Math processing error] it is most likely a caching problem with your browser. This may be solved by a “hard reload”, that is, by holding down the Shift key while clicking on the “reload/refresh” button. See, for example, this stackexchange query and reply.
The notation we use is that of Khovanskii (1963), which is a fundamental reference.
A continued fraction is a development of the form:
b0+a1b1+a2b2+a3b3+⋯a4b4
where the quantities a1,a2,… are called the partial numerators and b1,b2,… the partial denominators. If b0 as well as the partial numerators and denominators are all integers, with the partial denominators positive, then terminating the development at any stage leads to a result that may be written as a ratio of two integers, that is, as a vulgar fraction. If the termination is after n steps, we write the result as
PnQn=b0+a1b1+a2b2+a3b3+⋯1bn−1+anbn
Put P0/Q0=b0/1. The ratios Pn/Qn,n=0,1,2,… are called the convergents of the continued fraction (whether they converge in the mathematical sense or not).
It is convenient to extend the series of convergents artificially further back one step and put:
P−1Q−1=10,P0Q0=b01,P1Q1,P2Q2,⋯
With this definition, an easy inductive argument (Khovanskii 1963, 3) shows that the numerators and denominators of the convergents may be calculated progressively using a two-term recurrence relation. In fact the numerators and denominators have the same recurrence relations, namely:
Pn+1=bn+1Pn+an+1Pn−1Qn+1=bn+1Qn+an+1Qn−1}n=0,1,2,…
but of course Pn and Qn have different starting values.
A rational approximation to a real number is an approximation in the form of the ratio of two integers, numerator and denominator. The denominator should be positive, and for definiteness we require the numerator and denominator to be relatively prime, that is with greatest common divisor equal to 1.
The simplest and most powerful to find rational approximation for real numbers is to express them as continued fractions, with the convergents forming the approximations. An algorithm to do this is most easily described as a series of steps.
Let x be a real number for which the approximation is wanted.
Write b0=⌊x⌋ and put x=b0+(x−⌊x⌋)=b0+r0.
0≤r0<1, by definition. There are two cases:
If r0=0 the process is complete. The rational approximation is exact.
If r0>0, note that 1/r0>1. Write 1/r0=⌊1/r0⌋+(1/r0−⌊1/r0⌋)=b1+r1, with b1≥1 and 0≤r1<1. Then: x=b0+11/r0=b0+1b1+r1
Continuing in this way we produce a continued fraction expansion for the real number of the form:
x=b0+1b1+1b2+1b3+⋯1b4
where the partial numerators are all equal to 1 and the partial denominators are all positive integers. The fraction terminates if at any stage a ‘excess’ term, rn, becomes 0, preventing the process from continuing.
Continued fractions with all ai=1,i=1,2,… are usually called simple.
In some special cases the process can be conducted theoretically, yielding the entire fraction. For example the “golden ratio”, φ=(√5+1)/2 is the positive root of the equation 1/φ=φ−1. Writing this as φ−1=1/{1+(φ−1)} and iterating the right hand side leads directly to possibly the simplest continued fraction of the above form:
φ=1+11+11+11+⋯11
Further, the recurrence relations for the Pn and Qn are out-of-step Fibonacci sequence relations. This shows the well-known result that φ is the limit of the ratio of consecutive Fibonacci numbers:
F <- c(1, 1, numeric(15))
for(i in 3:17) ## Fibonacci sequence by recurrence
F[i] <- F[i-1] + F[i-2]
F
[1] 1 1 2 3 5 8 13 21 34 55 89 144 233 377
[15] 610 987 1597
vfractional((sqrt(5) + 1)/2, eps = 0, maxConv = 1:16)
[1] 1 2 3/2 5/3 8/5 13/8 21/13
[8] 34/21 55/34 89/55 144/89 233/144 377/233 610/377
[15] 987/610 1597/987
In a similar way we may write:
(√2−1)=√2−11×1+√21+√2=11+√2=12+(√2−1)
Iterating this equation in the same way, and slightly re-arranging, leads to the continued fraction:
√2=1+12+12+12+⋯12
One way to appreciate the algorithm is to look at how it might be coded in R
.
The package provides no function to return the partial denominators, bn, explicitly. However writing such a bespoke function is simple.
partial_denominators <- function(x, k = 10) {
b <- rep(NA, k)
r <- x
for(i in 1:k) {
b[i] <- floor(r)
r <- r - b[i]
if(isTRUE(all.equal(r, 0))) break
r <- 1/r
}
structure(b, names = paste0("b", 1:k-1))
}
To see it in action, we consider what it produces for the golden ratio, φ, the circular ratio, π, the base of natural logarithms, e, and the square roots of the first few positive integers. Irrational numbers have periodic continued fraction expansions, so the patterns will become clear.
x <- c(pi = base::pi, e = exp(1), phi = (sqrt(5) + 1)/2,
structure(sqrt(1:9), names = paste0("sqrt(", 1:9, ")")))
tab <- x %>% sapply(partial_denominators) %>% t
tab[is.na(tab)] <- ""
kable(tab, align = "r", caption = "Partial denominators")
b0 | b1 | b2 | b3 | b4 | b5 | b6 | b7 | b8 | b9 | |
---|---|---|---|---|---|---|---|---|---|---|
pi | 3 | 7 | 15 | 1 | 292 | 1 | 1 | 1 | 2 | 1 |
e | 2 | 1 | 2 | 1 | 1 | 4 | 1 | 1 | 6 | 1 |
phi | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 |
sqrt(1) | 1 | |||||||||
sqrt(2) | 1 | 2 | 2 | 2 | 2 | 2 | 2 | 2 | 2 | 2 |
sqrt(3) | 1 | 1 | 2 | 1 | 2 | 1 | 2 | 1 | 2 | 1 |
sqrt(4) | 2 | |||||||||
sqrt(5) | 2 | 4 | 4 | 4 | 4 | 4 | 4 | 4 | 4 | 4 |
sqrt(6) | 2 | 2 | 4 | 2 | 4 | 2 | 4 | 2 | 4 | 2 |
sqrt(7) | 2 | 1 | 1 | 1 | 4 | 1 | 1 | 1 | 4 | 1 |
sqrt(8) | 2 | 1 | 4 | 1 | 4 | 1 | 4 | 1 | 4 | 1 |
sqrt(9) | 3 |
That π and e do not appear to have any stable periodic pattern is not surprising, as they are known to be transcendental rather than merely irrational. The Euler constant e has a pattern of period 3, but with the third term increasing by two in each cycle. By contrast the denominators for π appear to be entirely random1.
These, and many others are listed in the On-line Encyclopedia of Integer Sequences, (OIES).
R
In R
the implementation is simple. The following routine computes the convergents (as pairs of integers), terminating either when the error in the rational approximation is below a set tolerance, or a prescribed maximum number of convergents is reached. The return value is a vector of the final three values: (Pn,Qn,n).
.ratr <- function(x, eps = 1.0e-6, maxConv = 20) {
PQ1 <- c(1, 0)
PQ2 <- c(floor(x), 1)
r <- x - PQ2[1]
i <- 0
while((i <- i+1) < maxConv && abs(x - PQ2[1]/PQ2[2]) > eps) {
b <- floor(1/r)
r <- 1/r - b
PQ0 <- PQ1
PQ1 <- PQ2
PQ2 <- b*PQ1 + PQ0
}
return(c(PQ2, i-1))
}
We can check the result with a well-known rational approximation:
pq <- .ratr(pi)
cat("Pn = ", pq[1], ", Qn = ", pq[2], ", n = ", pq[3], "\n", sep = "")
cat("pi = ", format(pi, digits = 15),
", Pn/Qn = ", format(pq[1]/pq[2], digits = 15),
", Error = ", pi - pq[1]/pq[2], "\n", sep = "")
Pn = 355, Qn = 113, n = 3
pi = 3.14159265358979, Pn/Qn = 3.14159292035398, Error = -0.0000002667642
The convergents for a rational approximation constructed in this way have some interesting elementary properties. Again, all results are taken from Chapter 1 of Khovanskii (1963).
For the convergents so obtained, Pn/Qn, the numerators and denominators will be relatively prime, that is, the fraction will be expressed in its lowest terms.
The even numbered convergents, P2n/Q2n form an increasing series of lower bounds to the true value and the odd numbered ones, P2n+1/Q2n+1 form a decreasing series of upper bounds: P0Q0<P2Q2<⋯<P2kQ2k<x<P2k+1Q2k+1<⋯<P3Q3<P1Q1 Hence the errors at any stage, x−Pn/Qn are alternately positive and negative.
The absolute error at any stage is bounded as follows: |x−PnQn|<1Qn−1Qn,n=1,2,…
From the recurrence relation, the denominators, Qn, are non-decreasing, ultimately increasing monotonically without limit, unless the continued fraction terminates after a finite number of convergents. It follows that the continued fraction either terminates at the true value, or converges in the limit to it. In practice, convergence is rapid.
We can illustrate some of these properties by looking at the sequence of convergents the process generates for π. Notice how the errors alternate in sign and rapidly decrease in absolute value. For reference, the accurate value of π is 3.141592653589793116
.
n | Pn/Qn | value | error |
---|---|---|---|
0 | 3 | 3.00000000000000 | 0.141592653589793 |
1 | 22/7 | 3.14285714285714 | -0.001264489267350 |
2 | 333/106 | 3.14150943396226 | 0.000083219627529 |
3 | 355/113 | 3.14159292035398 | -0.000000266764189 |
4 | 103993/33102 | 3.14159265301190 | 0.000000000577891 |
5 | 104348/33215 | 3.14159265392142 | -0.000000000331628 |
6 | 208341/66317 | 3.14159265346744 | 0.000000000122356 |
7 | 312689/99532 | 3.14159265361894 | -0.000000000029143 |
8 | 833719/265381 | 3.14159265358108 | 0.000000000008715 |
9 | 1146408/364913 | 3.14159265359140 | -0.000000000001611 |
Given a fixed denominator, d, the optimal rational approximation to x is clearly obtained by rounding xd, that is using ⌊xd+12⌋/d. The denominators Qn of the convergents usually correspond to points where there is a sudden increase in accuracy. To illustrate this we consider approximating √5. The first few convergents are as follows:
(s5 <- vfractional(sqrt(5), eps = 0, maxConv = 1:7))
[1] 2 9/4 38/17 161/72 682/305 2889/1292
[7] 12238/5473
d5 <- denominators(s5)
e5 <- abs(sqrt(5) - numerical(s5))
Now consider all rational approximations with denominators no larger:
d <- seq(max(d5))
n <- round(sqrt(5) * d)
To simplify the graph we may remove any which are not in their lowest terms.
gcd <- mapply(FUN = function(a, b) if(b == 0) a else Recall(b, a %% b),
n, d)
nd <- cbind(n, d)/gcd
nd <- nd[!duplicated(nd), ]
e <- abs(sqrt(5) - nd[, 1]/nd[, 2])
To see the detail of what is happening, we need to use a log-log scale:
The blue points give the errors in the approximations for all denominators up to 5473; the red points and lines indicate the subset of them which are the continued fraction convergents.
Note that the convergent, Pn/Qn, is the most accurate approximation for any with denominator less that or equal to Qn, and is closer than at most one or two with denominators less than Qn+1.
Khovanskii, Alexey Nikolaevitch. 1963. The Application of Continued Fractions and Their Generalizations to Problems in Approximation Theory. P. Noordhoff N. V.
π does have a regularly patterned continued fractions expression, but unlike e, it appears to have no such simple continued fraction.↩